/******/
(function (modules) { // webpackBootstrap
    /******/ 	// The module cache
    /******/
    var installedModules = {};

    /******/ 	// The require function
    /******/
    function __webpack_require__(moduleId) {

        /******/ 		// Check if module is in cache
        /******/
        if (installedModules[moduleId])
        /******/            return installedModules[moduleId].exports;

        /******/ 		// Create a new module (and put it into the cache)
        /******/
        var module = installedModules[moduleId] = {
            /******/            exports: {},
            /******/            id: moduleId,
            /******/            loaded: false
            /******/
        };

        /******/ 		// Execute the module function
        /******/
        modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);

        /******/ 		// Flag the module as loaded
        /******/
        module.loaded = true;

        /******/ 		// Return the exports of the module
        /******/
        return module.exports;
        /******/
    }


    /******/ 	// expose the modules object (__webpack_modules__)
    /******/
    __webpack_require__.m = modules;

    /******/ 	// expose the module cache
    /******/
    __webpack_require__.c = installedModules;

    /******/ 	// __webpack_public_path__
    /******/
    __webpack_require__.p = "./";

    /******/ 	// Load entry module and return exports
    /******/
    return __webpack_require__(0);
    /******/
})
/************************************************************************/
/******/({

    /***/ 0:
    /***/ function (module, exports, __webpack_require__) {

        module.exports = __webpack_require__(247);


        /***/
    },

    /***/ 183:
    /***/ function (module, exports, __webpack_require__) {

        var __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_DEFINE_RESULT__;//     Underscore.js 1.8.3
        //     http://underscorejs.org
        //     (c) 2009-2015 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
        //     Underscore may be freely distributed under the MIT license.

        (function () {

            // Baseline setup
            // --------------

            // Establish the root object, `window` in the browser, or `exports` on the server.
            var root = this;

            // Save the previous value of the `_` variable.
            var previousUnderscore = root._;

            // Save bytes in the minified (but not gzipped) version:
            var ArrayProto = Array.prototype, ObjProto = Object.prototype, FuncProto = Function.prototype;

            // Create quick reference variables for speed access to core prototypes.
            var
                push = ArrayProto.push,
                slice = ArrayProto.slice,
                toString = ObjProto.toString,
                hasOwnProperty = ObjProto.hasOwnProperty;

            // All **ECMAScript 5** native function implementations that we hope to use
            // are declared here.
            var
                nativeIsArray = Array.isArray,
                nativeKeys = Object.keys,
                nativeBind = FuncProto.bind,
                nativeCreate = Object.create;

            // Naked function reference for surrogate-prototype-swapping.
            var Ctor = function () {
            };

            // Create a safe reference to the Underscore object for use below.
            var _ = function (obj) {
                if (obj instanceof _) return obj;
                if (!(this instanceof _)) return new _(obj);
                this._wrapped = obj;
            };

            // Export the Underscore object for **Node.js**, with
            // backwards-compatibility for the old `require()` API. If we're in
            // the browser, add `_` as a global object.
            if (true) {
                if (typeof module !== 'undefined' && module.exports) {
                    exports = module.exports = _;
                }
                exports._ = _;
            } else {
                root._ = _;
            }

            // Current version.
            _.VERSION = '1.8.3';

            // Internal function that returns an efficient (for current engines) version
            // of the passed-in callback, to be repeatedly applied in other Underscore
            // functions.
            var optimizeCb = function (func, context, argCount) {
                if (context === void 0) return func;
                switch (argCount == null ? 3 : argCount) {
                    case 1:
                        return function (value) {
                            return func.call(context, value);
                        };
                    case 2:
                        return function (value, other) {
                            return func.call(context, value, other);
                        };
                    case 3:
                        return function (value, index, collection) {
                            return func.call(context, value, index, collection);
                        };
                    case 4:
                        return function (accumulator, value, index, collection) {
                            return func.call(context, accumulator, value, index, collection);
                        };
                }
                return function () {
                    return func.apply(context, arguments);
                };
            };

            // A mostly-internal function to generate callbacks that can be applied
            // to each element in a collection, returning the desired result — either
            // identity, an arbitrary callback, a property matcher, or a property accessor.
            var cb = function (value, context, argCount) {
                if (value == null) return _.identity;
                if (_.isFunction(value)) return optimizeCb(value, context, argCount);
                if (_.isObject(value)) return _.matcher(value);
                return _.property(value);
            };
            _.iteratee = function (value, context) {
                return cb(value, context, Infinity);
            };

            // An internal function for creating assigner functions.
            var createAssigner = function (keysFunc, undefinedOnly) {
                return function (obj) {
                    var length = arguments.length;
                    if (length < 2 || obj == null) return obj;
                    for (var index = 1; index < length; index++) {
                        var source = arguments[index],
                            keys = keysFunc(source),
                            l = keys.length;
                        for (var i = 0; i < l; i++) {
                            var key = keys[i];
                            if (!undefinedOnly || obj[key] === void 0) obj[key] = source[key];
                        }
                    }
                    return obj;
                };
            };

            // An internal function for creating a new object that inherits from another.
            var baseCreate = function (prototype) {
                if (!_.isObject(prototype)) return {};
                if (nativeCreate) return nativeCreate(prototype);
                Ctor.prototype = prototype;
                var result = new Ctor;
                Ctor.prototype = null;
                return result;
            };

            var property = function (key) {
                return function (obj) {
                    return obj == null ? void 0 : obj[key];
                };
            };

            // Helper for collection methods to determine whether a collection
            // should be iterated as an array or as an object
            // Related: http://people.mozilla.org/~jorendorff/es6-draft.html#sec-tolength
            // Avoids a very nasty iOS 8 JIT bug on ARM-64. #2094
            var MAX_ARRAY_INDEX = Math.pow(2, 53) - 1;
            var getLength = property('length');
            var isArrayLike = function (collection) {
                var length = getLength(collection);
                return typeof length == 'number' && length >= 0 && length <= MAX_ARRAY_INDEX;
            };

            // Collection Functions
            // --------------------

            // The cornerstone, an `each` implementation, aka `forEach`.
            // Handles raw objects in addition to array-likes. Treats all
            // sparse array-likes as if they were dense.
            _.each = _.forEach = function (obj, iteratee, context) {
                iteratee = optimizeCb(iteratee, context);
                var i, length;
                if (isArrayLike(obj)) {
                    for (i = 0, length = obj.length; i < length; i++) {
                        iteratee(obj[i], i, obj);
                    }
                } else {
                    var keys = _.keys(obj);
                    for (i = 0, length = keys.length; i < length; i++) {
                        iteratee(obj[keys[i]], keys[i], obj);
                    }
                }
                return obj;
            };

            // Return the results of applying the iteratee to each element.
            _.map = _.collect = function (obj, iteratee, context) {
                iteratee = cb(iteratee, context);
                var keys = !isArrayLike(obj) && _.keys(obj),
                    length = (keys || obj).length,
                    results = Array(length);
                for (var index = 0; index < length; index++) {
                    var currentKey = keys ? keys[index] : index;
                    results[index] = iteratee(obj[currentKey], currentKey, obj);
                }
                return results;
            };

            // Create a reducing function iterating left or right.
            function createReduce(dir) {
                // Optimized iterator function as using arguments.length
                // in the main function will deoptimize the, see #1991.
                function iterator(obj, iteratee, memo, keys, index, length) {
                    for (; index >= 0 && index < length; index += dir) {
                        var currentKey = keys ? keys[index] : index;
                        memo = iteratee(memo, obj[currentKey], currentKey, obj);
                    }
                    return memo;
                }

                return function (obj, iteratee, memo, context) {
                    iteratee = optimizeCb(iteratee, context, 4);
                    var keys = !isArrayLike(obj) && _.keys(obj),
                        length = (keys || obj).length,
                        index = dir > 0 ? 0 : length - 1;
                    // Determine the initial value if none is provided.
                    if (arguments.length < 3) {
                        memo = obj[keys ? keys[index] : index];
                        index += dir;
                    }
                    return iterator(obj, iteratee, memo, keys, index, length);
                };
            }

            // **Reduce** builds up a single result from a list of values, aka `inject`,
            // or `foldl`.
            _.reduce = _.foldl = _.inject = createReduce(1);

            // The right-associative version of reduce, also known as `foldr`.
            _.reduceRight = _.foldr = createReduce(-1);

            // Return the first value which passes a truth test. Aliased as `detect`.
            _.find = _.detect = function (obj, predicate, context) {
                var key;
                if (isArrayLike(obj)) {
                    key = _.findIndex(obj, predicate, context);
                } else {
                    key = _.findKey(obj, predicate, context);
                }
                if (key !== void 0 && key !== -1) return obj[key];
            };

            // Return all the elements that pass a truth test.
            // Aliased as `select`.
            _.filter = _.select = function (obj, predicate, context) {
                var results = [];
                predicate = cb(predicate, context);
                _.each(obj, function (value, index, list) {
                    if (predicate(value, index, list)) results.push(value);
                });
                return results;
            };

            // Return all the elements for which a truth test fails.
            _.reject = function (obj, predicate, context) {
                return _.filter(obj, _.negate(cb(predicate)), context);
            };

            // Determine whether all of the elements match a truth test.
            // Aliased as `all`.
            _.every = _.all = function (obj, predicate, context) {
                predicate = cb(predicate, context);
                var keys = !isArrayLike(obj) && _.keys(obj),
                    length = (keys || obj).length;
                for (var index = 0; index < length; index++) {
                    var currentKey = keys ? keys[index] : index;
                    if (!predicate(obj[currentKey], currentKey, obj)) return false;
                }
                return true;
            };

            // Determine if at least one element in the object matches a truth test.
            // Aliased as `any`.
            _.some = _.any = function (obj, predicate, context) {
                predicate = cb(predicate, context);
                var keys = !isArrayLike(obj) && _.keys(obj),
                    length = (keys || obj).length;
                for (var index = 0; index < length; index++) {
                    var currentKey = keys ? keys[index] : index;
                    if (predicate(obj[currentKey], currentKey, obj)) return true;
                }
                return false;
            };

            // Determine if the array or object contains a given item (using `===`).
            // Aliased as `includes` and `include`.
            _.contains = _.includes = _.include = function (obj, item, fromIndex, guard) {
                if (!isArrayLike(obj)) obj = _.values(obj);
                if (typeof fromIndex != 'number' || guard) fromIndex = 0;
                return _.indexOf(obj, item, fromIndex) >= 0;
            };

            // Invoke a method (with arguments) on every item in a collection.
            _.invoke = function (obj, method) {
                var args = slice.call(arguments, 2);
                var isFunc = _.isFunction(method);
                return _.map(obj, function (value) {
                    var func = isFunc ? method : value[method];
                    return func == null ? func : func.apply(value, args);
                });
            };

            // Convenience version of a common use case of `map`: fetching a property.
            _.pluck = function (obj, key) {
                return _.map(obj, _.property(key));
            };

            // Convenience version of a common use case of `filter`: selecting only objects
            // containing specific `key:value` pairs.
            _.where = function (obj, attrs) {
                return _.filter(obj, _.matcher(attrs));
            };

            // Convenience version of a common use case of `find`: getting the first object
            // containing specific `key:value` pairs.
            _.findWhere = function (obj, attrs) {
                return _.find(obj, _.matcher(attrs));
            };

            // Return the maximum element (or element-based computation).
            _.max = function (obj, iteratee, context) {
                var result = -Infinity, lastComputed = -Infinity,
                    value, computed;
                if (iteratee == null && obj != null) {
                    obj = isArrayLike(obj) ? obj : _.values(obj);
                    for (var i = 0, length = obj.length; i < length; i++) {
                        value = obj[i];
                        if (value > result) {
                            result = value;
                        }
                    }
                } else {
                    iteratee = cb(iteratee, context);
                    _.each(obj, function (value, index, list) {
                        computed = iteratee(value, index, list);
                        if (computed > lastComputed || computed === -Infinity && result === -Infinity) {
                            result = value;
                            lastComputed = computed;
                        }
                    });
                }
                return result;
            };

            // Return the minimum element (or element-based computation).
            _.min = function (obj, iteratee, context) {
                var result = Infinity, lastComputed = Infinity,
                    value, computed;
                if (iteratee == null && obj != null) {
                    obj = isArrayLike(obj) ? obj : _.values(obj);
                    for (var i = 0, length = obj.length; i < length; i++) {
                        value = obj[i];
                        if (value < result) {
                            result = value;
                        }
                    }
                } else {
                    iteratee = cb(iteratee, context);
                    _.each(obj, function (value, index, list) {
                        computed = iteratee(value, index, list);
                        if (computed < lastComputed || computed === Infinity && result === Infinity) {
                            result = value;
                            lastComputed = computed;
                        }
                    });
                }
                return result;
            };

            // Shuffle a collection, using the modern version of the
            // [Fisher-Yates shuffle](http://en.wikipedia.org/wiki/Fisher–Yates_shuffle).
            _.shuffle = function (obj) {
                var set = isArrayLike(obj) ? obj : _.values(obj);
                var length = set.length;
                var shuffled = Array(length);
                for (var index = 0, rand; index < length; index++) {
                    rand = _.random(0, index);
                    if (rand !== index) shuffled[index] = shuffled[rand];
                    shuffled[rand] = set[index];
                }
                return shuffled;
            };

            // Sample **n** random values from a collection.
            // If **n** is not specified, returns a single random element.
            // The internal `guard` argument allows it to work with `map`.
            _.sample = function (obj, n, guard) {
                if (n == null || guard) {
                    if (!isArrayLike(obj)) obj = _.values(obj);
                    return obj[_.random(obj.length - 1)];
                }
                return _.shuffle(obj).slice(0, Math.max(0, n));
            };

            // Sort the object's values by a criterion produced by an iteratee.
            _.sortBy = function (obj, iteratee, context) {
                iteratee = cb(iteratee, context);
                return _.pluck(_.map(obj, function (value, index, list) {
                    return {
                        value: value,
                        index: index,
                        criteria: iteratee(value, index, list)
                    };
                }).sort(function (left, right) {
                    var a = left.criteria;
                    var b = right.criteria;
                    if (a !== b) {
                        if (a > b || a === void 0) return 1;
                        if (a < b || b === void 0) return -1;
                    }
                    return left.index - right.index;
                }), 'value');
            };

            // An internal function used for aggregate "group by" operations.
            var group = function (behavior) {
                return function (obj, iteratee, context) {
                    var result = {};
                    iteratee = cb(iteratee, context);
                    _.each(obj, function (value, index) {
                        var key = iteratee(value, index, obj);
                        behavior(result, value, key);
                    });
                    return result;
                };
            };

            // Groups the object's values by a criterion. Pass either a string attribute
            // to group by, or a function that returns the criterion.
            _.groupBy = group(function (result, value, key) {
                if (_.has(result, key)) result[key].push(value); else result[key] = [value];
            });

            // Indexes the object's values by a criterion, similar to `groupBy`, but for
            // when you know that your index values will be unique.
            _.indexBy = group(function (result, value, key) {
                result[key] = value;
            });

            // Counts instances of an object that group by a certain criterion. Pass
            // either a string attribute to count by, or a function that returns the
            // criterion.
            _.countBy = group(function (result, value, key) {
                if (_.has(result, key)) result[key]++; else result[key] = 1;
            });

            // Safely create a real, live array from anything iterable.
            _.toArray = function (obj) {
                if (!obj) return [];
                if (_.isArray(obj)) return slice.call(obj);
                if (isArrayLike(obj)) return _.map(obj, _.identity);
                return _.values(obj);
            };

            // Return the number of elements in an object.
            _.size = function (obj) {
                if (obj == null) return 0;
                return isArrayLike(obj) ? obj.length : _.keys(obj).length;
            };

            // Split a collection into two arrays: one whose elements all satisfy the given
            // predicate, and one whose elements all do not satisfy the predicate.
            _.partition = function (obj, predicate, context) {
                predicate = cb(predicate, context);
                var pass = [], fail = [];
                _.each(obj, function (value, key, obj) {
                    (predicate(value, key, obj) ? pass : fail).push(value);
                });
                return [pass, fail];
            };

            // Array Functions
            // ---------------

            // Get the first element of an array. Passing **n** will return the first N
            // values in the array. Aliased as `head` and `take`. The **guard** check
            // allows it to work with `_.map`.
            _.first = _.head = _.take = function (array, n, guard) {
                if (array == null) return void 0;
                if (n == null || guard) return array[0];
                return _.initial(array, array.length - n);
            };

            // Returns everything but the last entry of the array. Especially useful on
            // the arguments object. Passing **n** will return all the values in
            // the array, excluding the last N.
            _.initial = function (array, n, guard) {
                return slice.call(array, 0, Math.max(0, array.length - (n == null || guard ? 1 : n)));
            };

            // Get the last element of an array. Passing **n** will return the last N
            // values in the array.
            _.last = function (array, n, guard) {
                if (array == null) return void 0;
                if (n == null || guard) return array[array.length - 1];
                return _.rest(array, Math.max(0, array.length - n));
            };

            // Returns everything but the first entry of the array. Aliased as `tail` and `drop`.
            // Especially useful on the arguments object. Passing an **n** will return
            // the rest N values in the array.
            _.rest = _.tail = _.drop = function (array, n, guard) {
                return slice.call(array, n == null || guard ? 1 : n);
            };

            // Trim out all falsy values from an array.
            _.compact = function (array) {
                return _.filter(array, _.identity);
            };

            // Internal implementation of a recursive `flatten` function.
            var flatten = function (input, shallow, strict, startIndex) {
                var output = [], idx = 0;
                for (var i = startIndex || 0, length = getLength(input); i < length; i++) {
                    var value = input[i];
                    if (isArrayLike(value) && (_.isArray(value) || _.isArguments(value))) {
                        //flatten current level of array or arguments object
                        if (!shallow) value = flatten(value, shallow, strict);
                        var j = 0, len = value.length;
                        output.length += len;
                        while (j < len) {
                            output[idx++] = value[j++];
                        }
                    } else if (!strict) {
                        output[idx++] = value;
                    }
                }
                return output;
            };

            // Flatten out an array, either recursively (by default), or just one level.
            _.flatten = function (array, shallow) {
                return flatten(array, shallow, false);
            };

            // Return a version of the array that does not contain the specified value(s).
            _.without = function (array) {
                return _.difference(array, slice.call(arguments, 1));
            };

            // Produce a duplicate-free version of the array. If the array has already
            // been sorted, you have the option of using a faster algorithm.
            // Aliased as `unique`.
            _.uniq = _.unique = function (array, isSorted, iteratee, context) {
                if (!_.isBoolean(isSorted)) {
                    context = iteratee;
                    iteratee = isSorted;
                    isSorted = false;
                }
                if (iteratee != null) iteratee = cb(iteratee, context);
                var result = [];
                var seen = [];
                for (var i = 0, length = getLength(array); i < length; i++) {
                    var value = array[i],
                        computed = iteratee ? iteratee(value, i, array) : value;
                    if (isSorted) {
                        if (!i || seen !== computed) result.push(value);
                        seen = computed;
                    } else if (iteratee) {
                        if (!_.contains(seen, computed)) {
                            seen.push(computed);
                            result.push(value);
                        }
                    } else if (!_.contains(result, value)) {
                        result.push(value);
                    }
                }
                return result;
            };

            // Produce an array that contains the union: each distinct element from all of
            // the passed-in arrays.
            _.union = function () {
                return _.uniq(flatten(arguments, true, true));
            };

            // Produce an array that contains every item shared between all the
            // passed-in arrays.
            _.intersection = function (array) {
                var result = [];
                var argsLength = arguments.length;
                for (var i = 0, length = getLength(array); i < length; i++) {
                    var item = array[i];
                    if (_.contains(result, item)) continue;
                    for (var j = 1; j < argsLength; j++) {
                        if (!_.contains(arguments[j], item)) break;
                    }
                    if (j === argsLength) result.push(item);
                }
                return result;
            };

            // Take the difference between one array and a number of other arrays.
            // Only the elements present in just the first array will remain.
            _.difference = function (array) {
                var rest = flatten(arguments, true, true, 1);
                return _.filter(array, function (value) {
                    return !_.contains(rest, value);
                });
            };

            // Zip together multiple lists into a single array -- elements that share
            // an index go together.
            _.zip = function () {
                return _.unzip(arguments);
            };

            // Complement of _.zip. Unzip accepts an array of arrays and groups
            // each array's elements on shared indices
            _.unzip = function (array) {
                var length = array && _.max(array, getLength).length || 0;
                var result = Array(length);

                for (var index = 0; index < length; index++) {
                    result[index] = _.pluck(array, index);
                }
                return result;
            };

            // Converts lists into objects. Pass either a single array of `[key, value]`
            // pairs, or two parallel arrays of the same length -- one of keys, and one of
            // the corresponding values.
            _.object = function (list, values) {
                var result = {};
                for (var i = 0, length = getLength(list); i < length; i++) {
                    if (values) {
                        result[list[i]] = values[i];
                    } else {
                        result[list[i][0]] = list[i][1];
                    }
                }
                return result;
            };

            // Generator function to create the findIndex and findLastIndex functions
            function createPredicateIndexFinder(dir) {
                return function (array, predicate, context) {
                    predicate = cb(predicate, context);
                    var length = getLength(array);
                    var index = dir > 0 ? 0 : length - 1;
                    for (; index >= 0 && index < length; index += dir) {
                        if (predicate(array[index], index, array)) return index;
                    }
                    return -1;
                };
            }

            // Returns the first index on an array-like that passes a predicate test
            _.findIndex = createPredicateIndexFinder(1);
            _.findLastIndex = createPredicateIndexFinder(-1);

            // Use a comparator function to figure out the smallest index at which
            // an object should be inserted so as to maintain order. Uses binary search.
            _.sortedIndex = function (array, obj, iteratee, context) {
                iteratee = cb(iteratee, context, 1);
                var value = iteratee(obj);
                var low = 0, high = getLength(array);
                while (low < high) {
                    var mid = Math.floor((low + high) / 2);
                    if (iteratee(array[mid]) < value) low = mid + 1; else high = mid;
                }
                return low;
            };

            // Generator function to create the indexOf and lastIndexOf functions
            function createIndexFinder(dir, predicateFind, sortedIndex) {
                return function (array, item, idx) {
                    var i = 0, length = getLength(array);
                    if (typeof idx == 'number') {
                        if (dir > 0) {
                            i = idx >= 0 ? idx : Math.max(idx + length, i);
                        } else {
                            length = idx >= 0 ? Math.min(idx + 1, length) : idx + length + 1;
                        }
                    } else if (sortedIndex && idx && length) {
                        idx = sortedIndex(array, item);
                        return array[idx] === item ? idx : -1;
                    }
                    if (item !== item) {
                        idx = predicateFind(slice.call(array, i, length), _.isNaN);
                        return idx >= 0 ? idx + i : -1;
                    }
                    for (idx = dir > 0 ? i : length - 1; idx >= 0 && idx < length; idx += dir) {
                        if (array[idx] === item) return idx;
                    }
                    return -1;
                };
            }

            // Return the position of the first occurrence of an item in an array,
            // or -1 if the item is not included in the array.
            // If the array is large and already in sort order, pass `true`
            // for **isSorted** to use binary search.
            _.indexOf = createIndexFinder(1, _.findIndex, _.sortedIndex);
            _.lastIndexOf = createIndexFinder(-1, _.findLastIndex);

            // Generate an integer Array containing an arithmetic progression. A port of
            // the native Python `range()` function. See
            // [the Python documentation](http://docs.python.org/library/functions.html#range).
            _.range = function (start, stop, step) {
                if (stop == null) {
                    stop = start || 0;
                    start = 0;
                }
                step = step || 1;

                var length = Math.max(Math.ceil((stop - start) / step), 0);
                var range = Array(length);

                for (var idx = 0; idx < length; idx++, start += step) {
                    range[idx] = start;
                }

                return range;
            };

            // Function (ahem) Functions
            // ------------------

            // Determines whether to execute a function as a constructor
            // or a normal function with the provided arguments
            var executeBound = function (sourceFunc, boundFunc, context, callingContext, args) {
                if (!(callingContext instanceof boundFunc)) return sourceFunc.apply(context, args);
                var self = baseCreate(sourceFunc.prototype);
                var result = sourceFunc.apply(self, args);
                if (_.isObject(result)) return result;
                return self;
            };

            // Create a function bound to a given object (assigning `this`, and arguments,
            // optionally). Delegates to **ECMAScript 5**'s native `Function.bind` if
            // available.
            _.bind = function (func, context) {
                if (nativeBind && func.bind === nativeBind) return nativeBind.apply(func, slice.call(arguments, 1));
                if (!_.isFunction(func)) throw new TypeError('Bind must be called on a function');
                var args = slice.call(arguments, 2);
                var bound = function () {
                    return executeBound(func, bound, context, this, args.concat(slice.call(arguments)));
                };
                return bound;
            };

            // Partially apply a function by creating a version that has had some of its
            // arguments pre-filled, without changing its dynamic `this` context. _ acts
            // as a placeholder, allowing any combination of arguments to be pre-filled.
            _.partial = function (func) {
                var boundArgs = slice.call(arguments, 1);
                var bound = function () {
                    var position = 0, length = boundArgs.length;
                    var args = Array(length);
                    for (var i = 0; i < length; i++) {
                        args[i] = boundArgs[i] === _ ? arguments[position++] : boundArgs[i];
                    }
                    while (position < arguments.length) args.push(arguments[position++]);
                    return executeBound(func, bound, this, this, args);
                };
                return bound;
            };

            // Bind a number of an object's methods to that object. Remaining arguments
            // are the method names to be bound. Useful for ensuring that all callbacks
            // defined on an object belong to it.
            _.bindAll = function (obj) {
                var i, length = arguments.length, key;
                if (length <= 1) throw new Error('bindAll must be passed function names');
                for (i = 1; i < length; i++) {
                    key = arguments[i];
                    obj[key] = _.bind(obj[key], obj);
                }
                return obj;
            };

            // Memoize an expensive function by storing its results.
            _.memoize = function (func, hasher) {
                var memoize = function (key) {
                    var cache = memoize.cache;
                    var address = '' + (hasher ? hasher.apply(this, arguments) : key);
                    if (!_.has(cache, address)) cache[address] = func.apply(this, arguments);
                    return cache[address];
                };
                memoize.cache = {};
                return memoize;
            };

            // Delays a function for the given number of milliseconds, and then calls
            // it with the arguments supplied.
            _.delay = function (func, wait) {
                var args = slice.call(arguments, 2);
                return setTimeout(function () {
                    return func.apply(null, args);
                }, wait);
            };

            // Defers a function, scheduling it to run after the current call stack has
            // cleared.
            _.defer = _.partial(_.delay, _, 1);

            // Returns a function, that, when invoked, will only be triggered at most once
            // during a given window of time. Normally, the throttled function will run
            // as much as it can, without ever going more than once per `wait` duration;
            // but if you'd like to disable the execution on the leading edge, pass
            // `{leading: false}`. To disable execution on the trailing edge, ditto.
            _.throttle = function (func, wait, options) {
                var context, args, result;
                var timeout = null;
                var previous = 0;
                if (!options) options = {};
                var later = function () {
                    previous = options.leading === false ? 0 : _.now();
                    timeout = null;
                    result = func.apply(context, args);
                    if (!timeout) context = args = null;
                };
                return function () {
                    var now = _.now();
                    if (!previous && options.leading === false) previous = now;
                    var remaining = wait - (now - previous);
                    context = this;
                    args = arguments;
                    if (remaining <= 0 || remaining > wait) {
                        if (timeout) {
                            clearTimeout(timeout);
                            timeout = null;
                        }
                        previous = now;
                        result = func.apply(context, args);
                        if (!timeout) context = args = null;
                    } else if (!timeout && options.trailing !== false) {
                        timeout = setTimeout(later, remaining);
                    }
                    return result;
                };
            };

            // Returns a function, that, as long as it continues to be invoked, will not
            // be triggered. The function will be called after it stops being called for
            // N milliseconds. If `immediate` is passed, trigger the function on the
            // leading edge, instead of the trailing.
            _.debounce = function (func, wait, immediate) {
                var timeout, args, context, timestamp, result;

                var later = function () {
                    var last = _.now() - timestamp;

                    if (last < wait && last >= 0) {
                        timeout = setTimeout(later, wait - last);
                    } else {
                        timeout = null;
                        if (!immediate) {
                            result = func.apply(context, args);
                            if (!timeout) context = args = null;
                        }
                    }
                };

                return function () {
                    context = this;
                    args = arguments;
                    timestamp = _.now();
                    var callNow = immediate && !timeout;
                    if (!timeout) timeout = setTimeout(later, wait);
                    if (callNow) {
                        result = func.apply(context, args);
                        context = args = null;
                    }

                    return result;
                };
            };

            // Returns the first function passed as an argument to the second,
            // allowing you to adjust arguments, run code before and after, and
            // conditionally execute the original function.
            _.wrap = function (func, wrapper) {
                return _.partial(wrapper, func);
            };

            // Returns a negated version of the passed-in predicate.
            _.negate = function (predicate) {
                return function () {
                    return !predicate.apply(this, arguments);
                };
            };

            // Returns a function that is the composition of a list of functions, each
            // consuming the return value of the function that follows.
            _.compose = function () {
                var args = arguments;
                var start = args.length - 1;
                return function () {
                    var i = start;
                    var result = args[start].apply(this, arguments);
                    while (i--) result = args[i].call(this, result);
                    return result;
                };
            };

            // Returns a function that will only be executed on and after the Nth call.
            _.after = function (times, func) {
                return function () {
                    if (--times < 1) {
                        return func.apply(this, arguments);
                    }
                };
            };

            // Returns a function that will only be executed up to (but not including) the Nth call.
            _.before = function (times, func) {
                var memo;
                return function () {
                    if (--times > 0) {
                        memo = func.apply(this, arguments);
                    }
                    if (times <= 1) func = null;
                    return memo;
                };
            };

            // Returns a function that will be executed at most one time, no matter how
            // often you call it. Useful for lazy initialization.
            _.once = _.partial(_.before, 2);

            // Object Functions
            // ----------------

            // Keys in IE < 9 that won't be iterated by `for key in ...` and thus missed.
            var hasEnumBug = !{toString: null}.propertyIsEnumerable('toString');
            var nonEnumerableProps = ['valueOf', 'isPrototypeOf', 'toString',
                'propertyIsEnumerable', 'hasOwnProperty', 'toLocaleString'];

            function collectNonEnumProps(obj, keys) {
                var nonEnumIdx = nonEnumerableProps.length;
                var constructor = obj.constructor;
                var proto = (_.isFunction(constructor) && constructor.prototype) || ObjProto;

                // Constructor is a special case.
                var prop = 'constructor';
                if (_.has(obj, prop) && !_.contains(keys, prop)) keys.push(prop);

                while (nonEnumIdx--) {
                    prop = nonEnumerableProps[nonEnumIdx];
                    if (prop in obj && obj[prop] !== proto[prop] && !_.contains(keys, prop)) {
                        keys.push(prop);
                    }
                }
            }

            // Retrieve the names of an object's own properties.
            // Delegates to **ECMAScript 5**'s native `Object.keys`
            _.keys = function (obj) {
                if (!_.isObject(obj)) return [];
                if (nativeKeys) return nativeKeys(obj);
                var keys = [];
                for (var key in obj) if (_.has(obj, key)) keys.push(key);
                // Ahem, IE < 9.
                if (hasEnumBug) collectNonEnumProps(obj, keys);
                return keys;
            };

            // Retrieve all the property names of an object.
            _.allKeys = function (obj) {
                if (!_.isObject(obj)) return [];
                var keys = [];
                for (var key in obj) keys.push(key);
                // Ahem, IE < 9.
                if (hasEnumBug) collectNonEnumProps(obj, keys);
                return keys;
            };

            // Retrieve the values of an object's properties.
            _.values = function (obj) {
                var keys = _.keys(obj);
                var length = keys.length;
                var values = Array(length);
                for (var i = 0; i < length; i++) {
                    values[i] = obj[keys[i]];
                }
                return values;
            };

            // Returns the results of applying the iteratee to each element of the object
            // In contrast to _.map it returns an object
            _.mapObject = function (obj, iteratee, context) {
                iteratee = cb(iteratee, context);
                var keys = _.keys(obj),
                    length = keys.length,
                    results = {},
                    currentKey;
                for (var index = 0; index < length; index++) {
                    currentKey = keys[index];
                    results[currentKey] = iteratee(obj[currentKey], currentKey, obj);
                }
                return results;
            };

            // Convert an object into a list of `[key, value]` pairs.
            _.pairs = function (obj) {
                var keys = _.keys(obj);
                var length = keys.length;
                var pairs = Array(length);
                for (var i = 0; i < length; i++) {
                    pairs[i] = [keys[i], obj[keys[i]]];
                }
                return pairs;
            };

            // Invert the keys and values of an object. The values must be serializable.
            _.invert = function (obj) {
                var result = {};
                var keys = _.keys(obj);
                for (var i = 0, length = keys.length; i < length; i++) {
                    result[obj[keys[i]]] = keys[i];
                }
                return result;
            };

            // Return a sorted list of the function names available on the object.
            // Aliased as `methods`
            _.functions = _.methods = function (obj) {
                var names = [];
                for (var key in obj) {
                    if (_.isFunction(obj[key])) names.push(key);
                }
                return names.sort();
            };

            // Extend a given object with all the properties in passed-in object(s).
            _.extend = createAssigner(_.allKeys);

            // Assigns a given object with all the own properties in the passed-in object(s)
            // (https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object/assign)
            _.extendOwn = _.assign = createAssigner(_.keys);

            // Returns the first key on an object that passes a predicate test
            _.findKey = function (obj, predicate, context) {
                predicate = cb(predicate, context);
                var keys = _.keys(obj), key;
                for (var i = 0, length = keys.length; i < length; i++) {
                    key = keys[i];
                    if (predicate(obj[key], key, obj)) return key;
                }
            };

            // Return a copy of the object only containing the whitelisted properties.
            _.pick = function (object, oiteratee, context) {
                var result = {}, obj = object, iteratee, keys;
                if (obj == null) return result;
                if (_.isFunction(oiteratee)) {
                    keys = _.allKeys(obj);
                    iteratee = optimizeCb(oiteratee, context);
                } else {
                    keys = flatten(arguments, false, false, 1);
                    iteratee = function (value, key, obj) {
                        return key in obj;
                    };
                    obj = Object(obj);
                }
                for (var i = 0, length = keys.length; i < length; i++) {
                    var key = keys[i];
                    var value = obj[key];
                    if (iteratee(value, key, obj)) result[key] = value;
                }
                return result;
            };

            // Return a copy of the object without the blacklisted properties.
            _.omit = function (obj, iteratee, context) {
                if (_.isFunction(iteratee)) {
                    iteratee = _.negate(iteratee);
                } else {
                    var keys = _.map(flatten(arguments, false, false, 1), String);
                    iteratee = function (value, key) {
                        return !_.contains(keys, key);
                    };
                }
                return _.pick(obj, iteratee, context);
            };

            // Fill in a given object with default properties.
            _.defaults = createAssigner(_.allKeys, true);

            // Creates an object that inherits from the given prototype object.
            // If additional properties are provided then they will be added to the
            // created object.
            _.create = function (prototype, props) {
                var result = baseCreate(prototype);
                if (props) _.extendOwn(result, props);
                return result;
            };

            // Create a (shallow-cloned) duplicate of an object.
            _.clone = function (obj) {
                if (!_.isObject(obj)) return obj;
                return _.isArray(obj) ? obj.slice() : _.extend({}, obj);
            };

            // Invokes interceptor with the obj, and then returns obj.
            // The primary purpose of this method is to "tap into" a method chain, in
            // order to perform operations on intermediate results within the chain.
            _.tap = function (obj, interceptor) {
                interceptor(obj);
                return obj;
            };

            // Returns whether an object has a given set of `key:value` pairs.
            _.isMatch = function (object, attrs) {
                var keys = _.keys(attrs), length = keys.length;
                if (object == null) return !length;
                var obj = Object(object);
                for (var i = 0; i < length; i++) {
                    var key = keys[i];
                    if (attrs[key] !== obj[key] || !(key in obj)) return false;
                }
                return true;
            };


            // Internal recursive comparison function for `isEqual`.
            var eq = function (a, b, aStack, bStack) {
                // Identical objects are equal. `0 === -0`, but they aren't identical.
                // See the [Harmony `egal` proposal](http://wiki.ecmascript.org/doku.php?id=harmony:egal).
                if (a === b) return a !== 0 || 1 / a === 1 / b;
                // A strict comparison is necessary because `null == undefined`.
                if (a == null || b == null) return a === b;
                // Unwrap any wrapped objects.
                if (a instanceof _) a = a._wrapped;
                if (b instanceof _) b = b._wrapped;
                // Compare `[[Class]]` names.
                var className = toString.call(a);
                if (className !== toString.call(b)) return false;
                switch (className) {
                    // Strings, numbers, regular expressions, dates, and booleans are compared by value.
                    case '[object RegExp]':
                    // RegExps are coerced to strings for comparison (Note: '' + /a/i === '/a/i')
                    case '[object String]':
                        // Primitives and their corresponding object wrappers are equivalent; thus, `"5"` is
                        // equivalent to `new String("5")`.
                        return '' + a === '' + b;
                    case '[object Number]':
                        // `NaN`s are equivalent, but non-reflexive.
                        // Object(NaN) is equivalent to NaN
                        if (+a !== +a) return +b !== +b;
                        // An `egal` comparison is performed for other numeric values.
                        return +a === 0 ? 1 / +a === 1 / b : +a === +b;
                    case '[object Date]':
                    case '[object Boolean]':
                        // Coerce dates and booleans to numeric primitive values. Dates are compared by their
                        // millisecond representations. Note that invalid dates with millisecond representations
                        // of `NaN` are not equivalent.
                        return +a === +b;
                }

                var areArrays = className === '[object Array]';
                if (!areArrays) {
                    if (typeof a != 'object' || typeof b != 'object') return false;

                    // Objects with different constructors are not equivalent, but `Object`s or `Array`s
                    // from different frames are.
                    var aCtor = a.constructor, bCtor = b.constructor;
                    if (aCtor !== bCtor && !(_.isFunction(aCtor) && aCtor instanceof aCtor &&
                        _.isFunction(bCtor) && bCtor instanceof bCtor)
                        && ('constructor' in a && 'constructor' in b)) {
                        return false;
                    }
                }
                // Assume equality for cyclic structures. The algorithm for detecting cyclic
                // structures is adapted from ES 5.1 section 15.12.3, abstract operation `JO`.

                // Initializing stack of traversed objects.
                // It's done here since we only need them for objects and arrays comparison.
                aStack = aStack || [];
                bStack = bStack || [];
                var length = aStack.length;
                while (length--) {
                    // Linear search. Performance is inversely proportional to the number of
                    // unique nested structures.
                    if (aStack[length] === a) return bStack[length] === b;
                }

                // Add the first object to the stack of traversed objects.
                aStack.push(a);
                bStack.push(b);

                // Recursively compare objects and arrays.
                if (areArrays) {
                    // Compare array lengths to determine if a deep comparison is necessary.
                    length = a.length;
                    if (length !== b.length) return false;
                    // Deep compare the contents, ignoring non-numeric properties.
                    while (length--) {
                        if (!eq(a[length], b[length], aStack, bStack)) return false;
                    }
                } else {
                    // Deep compare objects.
                    var keys = _.keys(a), key;
                    length = keys.length;
                    // Ensure that both objects contain the same number of properties before comparing deep equality.
                    if (_.keys(b).length !== length) return false;
                    while (length--) {
                        // Deep compare each member
                        key = keys[length];
                        if (!(_.has(b, key) && eq(a[key], b[key], aStack, bStack))) return false;
                    }
                }
                // Remove the first object from the stack of traversed objects.
                aStack.pop();
                bStack.pop();
                return true;
            };

            // Perform a deep comparison to check if two objects are equal.
            _.isEqual = function (a, b) {
                return eq(a, b);
            };

            // Is a given array, string, or object empty?
            // An "empty" object has no enumerable own-properties.
            _.isEmpty = function (obj) {
                if (obj == null) return true;
                if (isArrayLike(obj) && (_.isArray(obj) || _.isString(obj) || _.isArguments(obj))) return obj.length === 0;
                return _.keys(obj).length === 0;
            };

            // Is a given value a DOM element?
            _.isElement = function (obj) {
                return !!(obj && obj.nodeType === 1);
            };

            // Is a given value an array?
            // Delegates to ECMA5's native Array.isArray
            _.isArray = nativeIsArray || function (obj) {
                return toString.call(obj) === '[object Array]';
            };

            // Is a given variable an object?
            _.isObject = function (obj) {
                var type = typeof obj;
                return type === 'function' || type === 'object' && !!obj;
            };

            // Add some isType methods: isArguments, isFunction, isString, isNumber, isDate, isRegExp, isError.
            _.each(['Arguments', 'Function', 'String', 'Number', 'Date', 'RegExp', 'Error'], function (name) {
                _['is' + name] = function (obj) {
                    return toString.call(obj) === '[object ' + name + ']';
                };
            });

            // Define a fallback version of the method in browsers (ahem, IE < 9), where
            // there isn't any inspectable "Arguments" type.
            if (!_.isArguments(arguments)) {
                _.isArguments = function (obj) {
                    return _.has(obj, 'callee');
                };
            }

            // Optimize `isFunction` if appropriate. Work around some typeof bugs in old v8,
            // IE 11 (#1621), and in Safari 8 (#1929).
            if (typeof /./ != 'function' && typeof Int8Array != 'object') {
                _.isFunction = function (obj) {
                    return typeof obj == 'function' || false;
                };
            }

            // Is a given object a finite number?
            _.isFinite = function (obj) {
                return isFinite(obj) && !isNaN(parseFloat(obj));
            };

            // Is the given value `NaN`? (NaN is the only number which does not equal itself).
            _.isNaN = function (obj) {
                return _.isNumber(obj) && obj !== +obj;
            };

            // Is a given value a boolean?
            _.isBoolean = function (obj) {
                return obj === true || obj === false || toString.call(obj) === '[object Boolean]';
            };

            // Is a given value equal to null?
            _.isNull = function (obj) {
                return obj === null;
            };

            // Is a given variable undefined?
            _.isUndefined = function (obj) {
                return obj === void 0;
            };

            // Shortcut function for checking if an object has a given property directly
            // on itself (in other words, not on a prototype).
            _.has = function (obj, key) {
                return obj != null && hasOwnProperty.call(obj, key);
            };

            // Utility Functions
            // -----------------

            // Run Underscore.js in *noConflict* mode, returning the `_` variable to its
            // previous owner. Returns a reference to the Underscore object.
            _.noConflict = function () {
                root._ = previousUnderscore;
                return this;
            };

            // Keep the identity function around for default iteratees.
            _.identity = function (value) {
                return value;
            };

            // Predicate-generating functions. Often useful outside of Underscore.
            _.constant = function (value) {
                return function () {
                    return value;
                };
            };

            _.noop = function () {
            };

            _.property = property;

            // Generates a function for a given object that returns a given property.
            _.propertyOf = function (obj) {
                return obj == null ? function () {
                } : function (key) {
                    return obj[key];
                };
            };

            // Returns a predicate for checking whether an object has a given set of
            // `key:value` pairs.
            _.matcher = _.matches = function (attrs) {
                attrs = _.extendOwn({}, attrs);
                return function (obj) {
                    return _.isMatch(obj, attrs);
                };
            };

            // Run a function **n** times.
            _.times = function (n, iteratee, context) {
                var accum = Array(Math.max(0, n));
                iteratee = optimizeCb(iteratee, context, 1);
                for (var i = 0; i < n; i++) accum[i] = iteratee(i);
                return accum;
            };

            // Return a random integer between min and max (inclusive).
            _.random = function (min, max) {
                if (max == null) {
                    max = min;
                    min = 0;
                }
                return min + Math.floor(Math.random() * (max - min + 1));
            };

            // A (possibly faster) way to get the current timestamp as an integer.
            _.now = Date.now || function () {
                return new Date().getTime();
            };

            // List of HTML entities for escaping.
            var escapeMap = {
                '&': '&amp;',
                '<': '&lt;',
                '>': '&gt;',
                '"': '&quot;',
                "'": '&#x27;',
                '`': '&#x60;'
            };
            var unescapeMap = _.invert(escapeMap);

            // Functions for escaping and unescaping strings to/from HTML interpolation.
            var createEscaper = function (map) {
                var escaper = function (match) {
                    return map[match];
                };
                // Regexes for identifying a key that needs to be escaped
                var source = '(?:' + _.keys(map).join('|') + ')';
                var testRegexp = RegExp(source);
                var replaceRegexp = RegExp(source, 'g');
                return function (string) {
                    string = string == null ? '' : '' + string;
                    return testRegexp.test(string) ? string.replace(replaceRegexp, escaper) : string;
                };
            };
            _.escape = createEscaper(escapeMap);
            _.unescape = createEscaper(unescapeMap);

            // If the value of the named `property` is a function then invoke it with the
            // `object` as context; otherwise, return it.
            _.result = function (object, property, fallback) {
                var value = object == null ? void 0 : object[property];
                if (value === void 0) {
                    value = fallback;
                }
                return _.isFunction(value) ? value.call(object) : value;
            };

            // Generate a unique integer id (unique within the entire client session).
            // Useful for temporary DOM ids.
            var idCounter = 0;
            _.uniqueId = function (prefix) {
                var id = ++idCounter + '';
                return prefix ? prefix + id : id;
            };

            // By default, Underscore uses ERB-style template delimiters, change the
            // following template settings to use alternative delimiters.
            _.templateSettings = {
                evaluate: /<%([\s\S]+?)%>/g,
                interpolate: /<%=([\s\S]+?)%>/g,
                escape: /<%-([\s\S]+?)%>/g
            };

            // When customizing `templateSettings`, if you don't want to define an
            // interpolation, evaluation or escaping regex, we need one that is
            // guaranteed not to match.
            var noMatch = /(.)^/;

            // Certain characters need to be escaped so that they can be put into a
            // string literal.
            var escapes = {
                "'": "'",
                '\\': '\\',
                '\r': 'r',
                '\n': 'n',
                '\u2028': 'u2028',
                '\u2029': 'u2029'
            };

            var escaper = /\\|'|\r|\n|\u2028|\u2029/g;

            var escapeChar = function (match) {
                return '\\' + escapes[match];
            };

            // JavaScript micro-templating, similar to John Resig's implementation.
            // Underscore templating handles arbitrary delimiters, preserves whitespace,
            // and correctly escapes quotes within interpolated code.
            // NB: `oldSettings` only exists for backwards compatibility.
            _.template = function (text, settings, oldSettings) {
                if (!settings && oldSettings) settings = oldSettings;
                settings = _.defaults({}, settings, _.templateSettings);

                // Combine delimiters into one regular expression via alternation.
                var matcher = RegExp([
                    (settings.escape || noMatch).source,
                    (settings.interpolate || noMatch).source,
                    (settings.evaluate || noMatch).source
                ].join('|') + '|$', 'g');

                // Compile the template source, escaping string literals appropriately.
                var index = 0;
                var source = "__p+='";
                text.replace(matcher, function (match, escape, interpolate, evaluate, offset) {
                    source += text.slice(index, offset).replace(escaper, escapeChar);
                    index = offset + match.length;

                    if (escape) {
                        source += "'+\n((__t=(" + escape + "))==null?'':_.escape(__t))+\n'";
                    } else if (interpolate) {
                        source += "'+\n((__t=(" + interpolate + "))==null?'':__t)+\n'";
                    } else if (evaluate) {
                        source += "';\n" + evaluate + "\n__p+='";
                    }

                    // Adobe VMs need the match returned to produce the correct offest.
                    return match;
                });
                source += "';\n";

                // If a variable is not specified, place data values in local scope.
                if (!settings.variable) source = 'with(obj||{}){\n' + source + '}\n';

                source = "var __t,__p='',__j=Array.prototype.join," +
                    "print=function(){__p+=__j.call(arguments,'');};\n" +
                    source + 'return __p;\n';

                try {
                    var render = new Function(settings.variable || 'obj', '_', source);
                } catch (e) {
                    e.source = source;
                    throw e;
                }

                var template = function (data) {
                    return render.call(this, data, _);
                };

                // Provide the compiled source as a convenience for precompilation.
                var argument = settings.variable || 'obj';
                template.source = 'function(' + argument + '){\n' + source + '}';

                return template;
            };

            // Add a "chain" function. Start chaining a wrapped Underscore object.
            _.chain = function (obj) {
                var instance = _(obj);
                instance._chain = true;
                return instance;
            };

            // OOP
            // ---------------
            // If Underscore is called as a function, it returns a wrapped object that
            // can be used OO-style. This wrapper holds altered versions of all the
            // underscore functions. Wrapped objects may be chained.

            // Helper function to continue chaining intermediate results.
            var result = function (instance, obj) {
                return instance._chain ? _(obj).chain() : obj;
            };

            // Add your own custom functions to the Underscore object.
            _.mixin = function (obj) {
                _.each(_.functions(obj), function (name) {
                    var func = _[name] = obj[name];
                    _.prototype[name] = function () {
                        var args = [this._wrapped];
                        push.apply(args, arguments);
                        return result(this, func.apply(_, args));
                    };
                });
            };

            // Add all of the Underscore functions to the wrapper object.
            _.mixin(_);

            // Add all mutator Array functions to the wrapper.
            _.each(['pop', 'push', 'reverse', 'shift', 'sort', 'splice', 'unshift'], function (name) {
                var method = ArrayProto[name];
                _.prototype[name] = function () {
                    var obj = this._wrapped;
                    method.apply(obj, arguments);
                    if ((name === 'shift' || name === 'splice') && obj.length === 0) delete obj[0];
                    return result(this, obj);
                };
            });

            // Add all accessor Array functions to the wrapper.
            _.each(['concat', 'join', 'slice'], function (name) {
                var method = ArrayProto[name];
                _.prototype[name] = function () {
                    return result(this, method.apply(this._wrapped, arguments));
                };
            });

            // Extracts the result from a wrapped and chained object.
            _.prototype.value = function () {
                return this._wrapped;
            };

            // Provide unwrapping proxy for some methods used in engine operations
            // such as arithmetic and JSON stringification.
            _.prototype.valueOf = _.prototype.toJSON = _.prototype.value;

            _.prototype.toString = function () {
                return '' + this._wrapped;
            };

            // AMD registration happens at the end for compatibility with AMD loaders
            // that may not enforce next-turn semantics on modules. Even though general
            // practice for AMD registration is to be anonymous, underscore registers
            // as a named module because, like jQuery, it is a base library that is
            // popular enough to be bundled in a third party lib, but not be part of
            // an AMD load request. Those cases could generate an error when an
            // anonymous define() is called outside of a loader request.
            if (true) {
                !(__WEBPACK_AMD_DEFINE_ARRAY__ = [], __WEBPACK_AMD_DEFINE_RESULT__ = function () {
                    return _;
                }.apply(exports, __WEBPACK_AMD_DEFINE_ARRAY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));
            }
        }.call(this));


        /***/
    },

    /***/ 206:
    /***/ function (module, exports, __webpack_require__) {

        'use strict';

        var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) {
            return typeof obj;
        } : function (obj) {
            return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj;
        };

        ;
        (function () {

            var EMPTYFN = function EMPTYFN() {
            };
            var _code = __webpack_require__(207).code;
            var WEBIM_FILESIZE_LIMIT = 10485760;

            var _createStandardXHR = function _createStandardXHR() {
                try {
                    return new window.XMLHttpRequest();
                } catch (e) {
                    return false;
                }
            };

            var _createActiveXHR = function _createActiveXHR() {
                try {
                    return new window.ActiveXObject('Microsoft.XMLHTTP');
                } catch (e) {
                    return false;
                }
            };

            var _xmlrequest = function _xmlrequest(crossDomain) {
                crossDomain = crossDomain || true;
                var temp = _createStandardXHR() || _createActiveXHR();

                if ('withCredentials' in temp) {
                    return temp;
                }
                if (!crossDomain) {
                    return temp;
                }
                if (typeof window.XDomainRequest === 'undefined') {
                    return temp;
                }
                var xhr = new XDomainRequest();
                xhr.readyState = 0;
                xhr.status = 100;
                xhr.onreadystatechange = EMPTYFN;
                xhr.onload = function () {
                    xhr.readyState = 4;
                    xhr.status = 200;

                    var xmlDoc = new ActiveXObject('Microsoft.XMLDOM');
                    xmlDoc.async = 'false';
                    xmlDoc.loadXML(xhr.responseText);
                    xhr.responseXML = xmlDoc;
                    xhr.response = xhr.responseText;
                    xhr.onreadystatechange();
                };
                xhr.ontimeout = xhr.onerror = function () {
                    xhr.readyState = 4;
                    xhr.status = 500;
                    xhr.onreadystatechange();
                };
                return xhr;
            };

            var _hasFlash = function () {
                if ('ActiveXObject' in window) {
                    try {
                        return new ActiveXObject('ShockwaveFlash.ShockwaveFlash');
                    } catch (ex) {
                        return 0;
                    }
                } else {
                    if (navigator.plugins && navigator.plugins.length > 0) {
                        return navigator.plugins['Shockwave Flash'];
                    }
                }
                return 0;
            }();

            var _tmpUtilXHR = _xmlrequest(),
                _hasFormData = typeof FormData !== 'undefined',
                _hasBlob = typeof Blob !== 'undefined',
                _isCanSetRequestHeader = _tmpUtilXHR.setRequestHeader || false,
                _hasOverrideMimeType = _tmpUtilXHR.overrideMimeType || false,
                _isCanUploadFileAsync = _isCanSetRequestHeader && _hasFormData,
                _isCanUploadFile = _isCanUploadFileAsync || _hasFlash,
                _isCanDownLoadFile = _isCanSetRequestHeader && (_hasBlob || _hasOverrideMimeType);

            if (!Object.keys) {
                Object.keys = function () {
                    'use strict';

                    var hasOwnProperty = Object.prototype.hasOwnProperty,
                        hasDontEnumBug = !{toString: null}.propertyIsEnumerable('toString'),
                        dontEnums = ['toString', 'toLocaleString', 'valueOf', 'hasOwnProperty', 'isPrototypeOf', 'propertyIsEnumerable', 'constructor'],
                        dontEnumsLength = dontEnums.length;

                    return function (obj) {
                        if ((typeof obj === 'undefined' ? 'undefined' : _typeof(obj)) !== 'object' && (typeof obj !== 'function' || obj === null)) {
                            throw new TypeError('Object.keys called on non-object');
                        }

                        var result = [],
                            prop,
                            i;

                        for (prop in obj) {
                            if (hasOwnProperty.call(obj, prop)) {
                                result.push(prop);
                            }
                        }

                        if (hasDontEnumBug) {
                            for (i = 0; i < dontEnumsLength; i++) {
                                if (hasOwnProperty.call(obj, dontEnums[i])) {
                                    result.push(dontEnums[i]);
                                }
                            }
                        }
                        return result;
                    };
                }();
            }

            var utils = {
                hasFormData: _hasFormData,

                hasBlob: _hasBlob,

                emptyfn: EMPTYFN,

                isCanSetRequestHeader: _isCanSetRequestHeader,

                hasOverrideMimeType: _hasOverrideMimeType,

                isCanUploadFileAsync: _isCanUploadFileAsync,

                isCanUploadFile: _isCanUploadFile,

                isCanDownLoadFile: _isCanDownLoadFile,

                isSupportWss: function () {
                    var notSupportList = [
                        //1: QQ browser X5 core
                        /MQQBrowser[\/]5([.]\d+)?\sTBS/

                        //2: etc.
                        //...
                    ];

                    if (!window.WebSocket) {
                        return false;
                    }

                    var ua = window.navigator.userAgent;
                    for (var i = 0, l = notSupportList.length; i < l; i++) {
                        if (notSupportList[i].test(ua)) {
                            return false;
                        }
                    }
                    return true;
                }(),

                getIEVersion: function () {
                    var ua = navigator.userAgent,
                        matches,
                        tridentMap = {'4': 8, '5': 9, '6': 10, '7': 11};

                    matches = ua.match(/MSIE (\d+)/i);

                    if (matches && matches[1]) {
                        return +matches[1];
                    }
                    matches = ua.match(/Trident\/(\d+)/i);
                    if (matches && matches[1]) {
                        return tridentMap[matches[1]] || null;
                    }
                    return null;
                }(),

                stringify: function stringify(json) {
                    if (typeof JSON !== 'undefined' && JSON.stringify) {
                        return JSON.stringify(json);
                    } else {
                        var s = '',
                            arr = [];

                        var iterate = function iterate(json) {
                            var isArr = false;

                            if (Object.prototype.toString.call(json) === '[object Array]') {
                                arr.push(']', '[');
                                isArr = true;
                            } else if (Object.prototype.toString.call(json) === '[object Object]') {
                                arr.push('}', '{');
                            }

                            for (var o in json) {
                                if (Object.prototype.toString.call(json[o]) === '[object Null]') {
                                    json[o] = 'null';
                                } else if (Object.prototype.toString.call(json[o]) === '[object Undefined]') {
                                    json[o] = 'undefined';
                                }

                                if (json[o] && _typeof(json[o]) === 'object') {
                                    s += ',' + (isArr ? '' : '"' + o + '":' + (isArr ? '"' : '')) + iterate(json[o]) + '';
                                } else {
                                    s += ',"' + (isArr ? '' : o + '":"') + json[o] + '"';
                                }
                            }

                            if (s != '') {
                                s = s.slice(1);
                            }

                            return arr.pop() + s + arr.pop();
                        };
                        return iterate(json);
                    }
                },
                login: function login(options) {
                    var options = options || {};
                    var suc = options.success || EMPTYFN;
                    var err = options.error || EMPTYFN;

                    var appKey = options.appKey || '';
                    var devInfos = appKey.split('#');
                    if (devInfos.length !== 2) {
                        err({
                            type: _code.WEBIM_CONNCTION_APPKEY_NOT_ASSIGN_ERROR
                        });
                        return false;
                    }

                    var orgName = devInfos[0];
                    var appName = devInfos[1];
                    var https = https || options.https;
                    var user = options.user || '';
                    var pwd = options.pwd || '';

                    var apiUrl = options.apiUrl;

                    var loginJson = {
                        grant_type: 'password',
                        username: user,
                        password: pwd,
                        timestamp: +new Date()
                    };
                    var loginfo = utils.stringify(loginJson);

                    var options = {
                        url: apiUrl + '/' + orgName + '/' + appName + '/token',
                        dataType: 'json',
                        data: loginfo,
                        success: suc,
                        error: err
                    };
                    return utils.ajax(options);
                },

                getFileUrl: function getFileUrl(fileInputId) {
                    var uri = {
                        url: '',
                        filename: '',
                        filetype: '',
                        data: ''
                    };

                    var fileObj = typeof fileInputId === 'string' ? document.getElementById(fileInputId) : fileInputId;

                    if (!utils.isCanUploadFileAsync || !fileObj) {
                        return uri;
                    }
                    try {
                        if (window.URL.createObjectURL) {
                            var fileItems = fileObj.files;
                            if (fileItems.length > 0) {
                                var u = fileItems.item(0);
                                uri.data = u;
                                uri.url = window.URL.createObjectURL(u);
                                uri.filename = u.name || '';
                            }
                        } else {
                            // IE
                            var u = document.getElementById(fileInputId).value;
                            uri.url = u;
                            var pos1 = u.lastIndexOf('/');
                            var pos2 = u.lastIndexOf('\\');
                            var pos = Math.max(pos1, pos2);
                            if (pos < 0) uri.filename = u; else uri.filename = u.substring(pos + 1);
                        }
                        var index = uri.filename.lastIndexOf('.');
                        if (index != -1) {
                            uri.filetype = uri.filename.substring(index + 1).toLowerCase();
                        }
                        return uri;
                    } catch (e) {
                        throw e;
                    }
                },

                getFileSize: function getFileSize(file) {
                    var fileSize = this.getFileLength(file);
                    if (fileSize > 10000000) {
                        return false;
                    }
                    var kb = Math.round(fileSize / 1000);
                    if (kb < 1000) {
                        fileSize = kb + ' KB';
                    } else if (kb >= 1000) {
                        var mb = kb / 1000;
                        if (mb < 1000) {
                            fileSize = mb.toFixed(1) + ' MB';
                        } else {
                            var gb = mb / 1000;
                            fileSize = gb.toFixed(1) + ' GB';
                        }
                    }
                    return fileSize;
                },

                getFileLength: function getFileLength(file) {
                    var fileLength = 0;
                    if (file) {
                        if (file.files) {
                            if (file.files.length > 0) {
                                fileLength = file.files[0].size;
                            }
                        } else if (file.select && 'ActiveXObject' in window) {
                            file.select();
                            var fileobject = new ActiveXObject('Scripting.FileSystemObject');
                            var file = fileobject.GetFile(file.value);
                            fileLength = file.Size;
                        }
                    }
                    return fileLength;
                },

                hasFlash: _hasFlash,

                trim: function trim(str) {

                    str = typeof str === 'string' ? str : '';

                    return str.trim ? str.trim() : str.replace(/^\s|\s$/g, '');
                },

                parseEmoji: function parseEmoji(msg) {
                    if (typeof WebIM.Emoji === 'undefined' || typeof WebIM.Emoji.map === 'undefined') {
                        return msg;
                    } else {
                        var emoji = WebIM.Emoji,
                            reg = null;

                        for (var face in emoji.map) {
                            if (emoji.map.hasOwnProperty(face)) {
                                while (msg.indexOf(face) > -1) {
                                    msg = msg.replace(face, '<img class="emoji" src="' + emoji.path + emoji.map[face] + '" />');
                                }
                            }
                        }
                        return msg;
                    }
                },

                parseLink: function parseLink(msg) {

                    var reg = /(https?\:\/\/|www\.)([a-zA-Z0-9-]+(\.[a-zA-Z0-9]+)+)(\:[0-9]{2,4})?\/?((\.[:_0-9a-zA-Z-]+)|[:_0-9a-zA-Z-]*\/?)*\??[:_#@*&%0-9a-zA-Z-/=]*/gm;

                    msg = msg.replace(reg, function (v) {

                        var prefix = /^https?/gm.test(v);

                        return "<a href='" + (prefix ? v : '//' + v) + "' target='_blank'>" + v + "</a>";
                    });

                    return msg;
                },

                parseJSON: function parseJSON(data) {

                    if (window.JSON && window.JSON.parse) {
                        return window.JSON.parse(data + '');
                    }

                    var requireNonComma,
                        depth = null,
                        str = utils.trim(data + '');

                    return str && !utils.trim(str.replace(/(,)|(\[|{)|(}|])|"(?:[^"\\\r\n]|\\["\\\/bfnrt]|\\u[\da-fA-F]{4})*"\s*:?|true|false|null|-?(?!0\d)\d+(?:\.\d+|)(?:[eE][+-]?\d+|)/g, function (token, comma, open, close) {

                        if (requireNonComma && comma) {
                            depth = 0;
                        }

                        if (depth === 0) {
                            return token;
                        }

                        requireNonComma = open || comma;
                        depth += !close - !open;
                        return '';
                    })) ? Function('return ' + str)() : Function('Invalid JSON: ' + data)();
                },

                parseUploadResponse: function parseUploadResponse(response) {
                    return response.indexOf('callback') > -1 ? //lte ie9
                        response.slice(9, -1) : response;
                },

                parseDownloadResponse: function parseDownloadResponse(response) {
                    return response && response.type && response.type === 'application/json' || 0 > Object.prototype.toString.call(response).indexOf('Blob') ? this.url + '?token=' : window.URL.createObjectURL(response);
                },

                uploadFile: function uploadFile(options) {
                    var options = options || {};
                    options.onFileUploadProgress = options.onFileUploadProgress || EMPTYFN;
                    options.onFileUploadComplete = options.onFileUploadComplete || EMPTYFN;
                    options.onFileUploadError = options.onFileUploadError || EMPTYFN;
                    options.onFileUploadCanceled = options.onFileUploadCanceled || EMPTYFN;

                    var acc = options.accessToken || this.context.accessToken;
                    if (!acc) {
                        options.onFileUploadError({
                            type: _code.WEBIM_UPLOADFILE_NO_LOGIN,
                            id: options.id
                        });
                        return;
                    }

                    var orgName, appName, devInfos;
                    var appKey = options.appKey || this.context.appKey || '';

                    if (appKey) {
                        devInfos = appKey.split('#');
                        orgName = devInfos[0];
                        appName = devInfos[1];
                    }

                    if (!orgName && !appName) {
                        options.onFileUploadError({
                            type: _code.WEBIM_UPLOADFILE_ERROR,
                            id: options.id
                        });
                        return;
                    }

                    var apiUrl = options.apiUrl;
                    var uploadUrl = apiUrl + '/' + orgName + '/' + appName + '/chatfiles';

                    if (!utils.isCanUploadFileAsync) {
                        if (utils.hasFlash && typeof options.flashUpload === 'function') {
                            options.flashUpload && options.flashUpload(uploadUrl, options);
                        } else {
                            options.onFileUploadError({
                                type: _code.WEBIM_UPLOADFILE_BROWSER_ERROR,
                                id: options.id
                            });
                        }
                        return;
                    }

                    var fileSize = options.file.data ? options.file.data.size : undefined;
                    if (fileSize > WEBIM_FILESIZE_LIMIT) {
                        options.onFileUploadError({
                            type: _code.WEBIM_UPLOADFILE_ERROR,
                            id: options.id
                        });
                        return;
                    } else if (fileSize <= 0) {
                        options.onFileUploadError({
                            type: _code.WEBIM_UPLOADFILE_ERROR,
                            id: options.id
                        });
                        return;
                    }

                    var xhr = utils.xmlrequest();
                    var onError = function onError(e) {
                        options.onFileUploadError({
                            type: _code.WEBIM_UPLOADFILE_ERROR,
                            id: options.id,
                            xhr: xhr
                        });
                    };
                    if (xhr.upload) {
                        xhr.upload.addEventListener('progress', options.onFileUploadProgress, false);
                    }
                    if (xhr.addEventListener) {
                        xhr.addEventListener('abort', options.onFileUploadCanceled, false);
                        xhr.addEventListener('load', function (e) {
                            try {
                                var json = utils.parseJSON(xhr.responseText);
                                try {
                                    options.onFileUploadComplete(json);
                                } catch (e) {
                                    options.onFileUploadError({
                                        type: _code.WEBIM_CONNCTION_CALLBACK_INNER_ERROR,
                                        data: e
                                    });
                                }
                            } catch (e) {
                                options.onFileUploadError({
                                    type: _code.WEBIM_UPLOADFILE_ERROR,
                                    data: xhr.responseText,
                                    id: options.id,
                                    xhr: xhr
                                });
                            }
                        }, false);
                        xhr.addEventListener('error', onError, false);
                    } else if (xhr.onreadystatechange) {
                        xhr.onreadystatechange = function () {
                            if (xhr.readyState === 4) {
                                if (ajax.status === 200) {
                                    try {
                                        var json = utils.parseJSON(xhr.responseText);
                                        options.onFileUploadComplete(json);
                                    } catch (e) {
                                        options.onFileUploadError({
                                            type: _code.WEBIM_UPLOADFILE_ERROR,
                                            data: xhr.responseText,
                                            id: options.id,
                                            xhr: xhr
                                        });
                                    }
                                } else {
                                    options.onFileUploadError({
                                        type: _code.WEBIM_UPLOADFILE_ERROR,
                                        data: xhr.responseText,
                                        id: options.id,
                                        xhr: xhr
                                    });
                                }
                            } else {
                                xhr.abort();
                                options.onFileUploadCanceled();
                            }
                        };
                    }

                    xhr.open('POST', uploadUrl);

                    xhr.setRequestHeader('restrict-access', 'true');
                    xhr.setRequestHeader('Accept', '*/*'); // Android QQ browser has some problem with this attribute.
                    xhr.setRequestHeader('Authorization', 'Bearer ' + acc);

                    var formData = new FormData();
                    formData.append('file', options.file.data);
                    // fix: ie8 status error
                    window.XDomainRequest && (xhr.readyState = 2);
                    xhr.send(formData);
                },

                download: function download(options) {
                    options.onFileDownloadComplete = options.onFileDownloadComplete || EMPTYFN;
                    options.onFileDownloadError = options.onFileDownloadError || EMPTYFN;

                    var accessToken = options.accessToken || this.context.accessToken;
                    if (!accessToken) {
                        options.onFileDownloadError({
                            type: _code.WEBIM_DOWNLOADFILE_NO_LOGIN,
                            id: options.id
                        });
                        return;
                    }

                    var onError = function onError(e) {
                        options.onFileDownloadError({
                            type: _code.WEBIM_DOWNLOADFILE_ERROR,
                            id: options.id,
                            xhr: xhr
                        });
                    };

                    if (!utils.isCanDownLoadFile) {
                        options.onFileDownloadComplete();
                        return;
                    }
                    var xhr = utils.xmlrequest();
                    if ('addEventListener' in xhr) {
                        xhr.addEventListener('load', function (e) {
                            options.onFileDownloadComplete(xhr.response, xhr);
                        }, false);
                        xhr.addEventListener('error', onError, false);
                    } else if ('onreadystatechange' in xhr) {
                        xhr.onreadystatechange = function () {
                            if (xhr.readyState === 4) {
                                if (ajax.status === 200) {
                                    options.onFileDownloadComplete(xhr.response, xhr);
                                } else {
                                    options.onFileDownloadError({
                                        type: _code.WEBIM_DOWNLOADFILE_ERROR,
                                        id: options.id,
                                        xhr: xhr
                                    });
                                }
                            } else {
                                xhr.abort();
                                options.onFileDownloadError({
                                    type: _code.WEBIM_DOWNLOADFILE_ERROR,
                                    id: options.id,
                                    xhr: xhr
                                });
                            }
                        };
                    }

                    var method = options.method || 'GET';
                    var resType = options.responseType || 'blob';
                    var mimeType = options.mimeType || 'text/plain; charset=x-user-defined';
                    xhr.open(method, options.url);
                    if (typeof Blob !== 'undefined') {
                        xhr.responseType = resType;
                    } else {
                        xhr.overrideMimeType(mimeType);
                    }

                    var innerHeaer = {
                        'X-Requested-With': 'XMLHttpRequest',
                        'Accept': 'application/octet-stream',
                        'share-secret': options.secret,
                        'Authorization': 'Bearer ' + accessToken
                    };
                    var headers = options.headers || {};
                    for (var key in headers) {
                        innerHeaer[key] = headers[key];
                    }
                    for (var key in innerHeaer) {
                        if (innerHeaer[key]) {
                            xhr.setRequestHeader(key, innerHeaer[key]);
                        }
                    }
                    // fix: ie8 status error
                    window.XDomainRequest && (xhr.readyState = 2);
                    xhr.send(null);
                },

                parseTextMessage: function parseTextMessage(message, faces) {
                    if (typeof message !== 'string') {
                        return;
                    }

                    if (Object.prototype.toString.call(faces) !== '[object Object]') {
                        return {
                            isemoji: false,
                            body: [{
                                type: 'txt',
                                data: message
                            }]
                        };
                    }

                    var receiveMsg = message;
                    var emessage = [];
                    var expr = /\[[^[\]]{2,3}\]/mg;
                    var emoji = receiveMsg.match(expr);

                    if (!emoji || emoji.length < 1) {
                        return {
                            isemoji: false,
                            body: [{
                                type: 'txt',
                                data: message
                            }]
                        };
                    }
                    var isemoji = false;
                    for (var i = 0; i < emoji.length; i++) {
                        var tmsg = receiveMsg.substring(0, receiveMsg.indexOf(emoji[i])),
                            existEmoji = WebIM.Emoji.map[emoji[i]];

                        if (tmsg) {
                            emessage.push({
                                type: 'txt',
                                data: tmsg
                            });
                        }
                        if (!existEmoji) {
                            emessage.push({
                                type: 'txt',
                                data: emoji[i]
                            });
                            continue;
                        }
                        var emojiStr = WebIM.Emoji.map ? WebIM.Emoji.path + existEmoji : null;

                        if (emojiStr) {
                            isemoji = true;
                            emessage.push({
                                type: 'emoji',
                                data: emojiStr
                            });
                        } else {
                            emessage.push({
                                type: 'txt',
                                data: emoji[i]
                            });
                        }
                        var restMsgIndex = receiveMsg.indexOf(emoji[i]) + emoji[i].length;
                        receiveMsg = receiveMsg.substring(restMsgIndex);
                    }
                    if (receiveMsg) {
                        emessage.push({
                            type: 'txt',
                            data: receiveMsg
                        });
                    }
                    if (isemoji) {
                        return {
                            isemoji: isemoji,
                            body: emessage
                        };
                    }
                    return {
                        isemoji: false,
                        body: [{
                            type: 'txt',
                            data: message
                        }]
                    };
                },

                parseUri: function parseUri() {
                    var pattern = /([^\?|&])\w+=([^&]+)/g;
                    var uri = {};
                    if (window.location.search) {
                        var args = window.location.search.match(pattern);
                        for (var i in args) {
                            var str = args[i];
                            var eq = str.indexOf('=');
                            var key = str.substr(0, eq);
                            var value = str.substr(eq + 1);
                            uri[key] = value;
                        }
                    }
                    return uri;
                },

                parseHrefHash: function parseHrefHash() {
                    var pattern = /([^\#|&])\w+=([^&]+)/g;
                    var uri = {};
                    if (window.location.hash) {
                        var args = window.location.hash.match(pattern);
                        for (var i in args) {
                            var str = args[i];
                            var eq = str.indexOf('=');
                            var key = str.substr(0, eq);
                            var value = str.substr(eq + 1);
                            uri[key] = value;
                        }
                    }
                    return uri;
                },

                xmlrequest: _xmlrequest,

                getXmlFirstChild: function getXmlFirstChild(data, tagName) {
                    var children = data.getElementsByTagName(tagName);
                    if (children.length == 0) {
                        return null;
                    } else {
                        return children[0];
                    }
                },
                ajax: function ajax(options) {
                    var dataType = options.dataType || 'text';
                    var suc = options.success || EMPTYFN;
                    var error = options.error || EMPTYFN;
                    var xhr = utils.xmlrequest();

                    xhr.onreadystatechange = function () {
                        if (xhr.readyState === 4) {
                            var status = xhr.status || 0;
                            if (status === 200) {
                                try {
                                    switch (dataType) {
                                        case 'text':
                                            suc(xhr.responseText);
                                            return;
                                        case 'json':
                                            var json = utils.parseJSON(xhr.responseText);
                                            suc(json, xhr);
                                            return;
                                        case 'xml':
                                            if (xhr.responseXML && xhr.responseXML.documentElement) {
                                                suc(xhr.responseXML.documentElement, xhr);
                                            } else {
                                                error({
                                                    type: _code.WEBIM_CONNCTION_AJAX_ERROR,
                                                    data: xhr.responseText
                                                });
                                            }
                                            return;
                                    }
                                    suc(xhr.response || xhr.responseText, xhr);
                                } catch (e) {
                                    error({
                                        type: _code.WEBIM_CONNCTION_AJAX_ERROR,
                                        data: e
                                    });
                                }
                                return;
                            } else {
                                error({
                                    type: _code.WEBIM_CONNCTION_AJAX_ERROR,
                                    data: xhr.responseText
                                });
                                return;
                            }
                        }
                        if (xhr.readyState === 0) {
                            error({
                                type: _code.WEBIM_CONNCTION_AJAX_ERROR,
                                data: xhr.responseText
                            });
                        }
                    };

                    if (options.responseType) {
                        if (xhr.responseType) {
                            xhr.responseType = options.responseType;
                        }
                    }
                    if (options.mimeType) {
                        if (utils.hasOverrideMimeType) {
                            xhr.overrideMimeType(options.mimeType);
                        }
                    }

                    var type = options.type || 'POST',
                        data = options.data || null,
                        tempData = '';

                    if (type.toLowerCase() === 'get' && data) {
                        for (var o in data) {
                            if (data.hasOwnProperty(o)) {
                                tempData += o + '=' + data[o] + '&';
                            }
                        }
                        tempData = tempData ? tempData.slice(0, -1) : tempData;
                        options.url += (options.url.indexOf('?') > 0 ? '&' : '?') + (tempData ? tempData + '&' : tempData) + '_v=' + new Date().getTime();
                        data = null;
                        tempData = null;
                    }
                    xhr.open(type, options.url, utils.isCanSetRequestHeader);

                    if (utils.isCanSetRequestHeader) {
                        var headers = options.headers || {};
                        for (var key in headers) {
                            if (headers.hasOwnProperty(key)) {
                                xhr.setRequestHeader(key, headers[key]);
                            }
                        }
                    }
                    // fix: ie8 status error
                    window.XDomainRequest && (xhr.readyState = 2);
                    xhr.send(data);
                    return xhr;
                },
                ts: function ts() {
                    var d = new Date();
                    var Hours = d.getHours(); //获取当前小时数(0-23)
                    var Minutes = d.getMinutes(); //获取当前分钟数(0-59)
                    var Seconds = d.getSeconds(); //获取当前秒数(0-59)
                    var Milliseconds = d.getMilliseconds(); //获取当前毫秒
                    return (Hours < 10 ? "0" + Hours : Hours) + ':' + (Minutes < 10 ? "0" + Minutes : Minutes) + ':' + (Seconds < 10 ? "0" + Seconds : Seconds) + ':' + Milliseconds + ' ';
                },

                getObjectKey: function getObjectKey(obj, val) {
                    for (var key in obj) {
                        if (obj[key] == val) {
                            return key;
                        }
                    }
                    return '';
                },

                sprintf: function sprintf() {
                    var arg = arguments,
                        str = arg[0] || '',
                        i,
                        len;
                    for (i = 1, len = arg.length; i < len; i++) {
                        str = str.replace(/%s/, arg[i]);
                    }
                    return str;
                },

                setCookie: function setCookie(name, value, days) {
                    var cookie = name + '=' + encodeURIComponent(value);
                    if (typeof days == 'number') {
                        cookie += '; max-age: ' + days * 60 * 60 * 24;
                    }
                    document.cookie = cookie;
                },

                getCookie: function getCookie() {
                    var allCookie = {};
                    var all = document.cookie;
                    if (all === "") {
                        return allCookie;
                    }
                    var list = all.split("; ");
                    for (var i = 0; i < list.length; i++) {
                        var cookie = list[i];
                        var p = cookie.indexOf('=');
                        var name = cookie.substring(0, p);
                        var value = cookie.substring(p + 1);
                        value = decodeURIComponent(value);
                        allCookie[name] = value;
                    }
                    return allCookie;
                }
            };

            exports.utils = utils;
        })();

        /***/
    },

    /***/ 207:
    /***/ function (module, exports) {

        "use strict";

        ;
        (function () {

            exports.code = {
                WEBIM_CONNCTION_USER_NOT_ASSIGN_ERROR: 0,
                WEBIM_CONNCTION_OPEN_ERROR: 1,
                WEBIM_CONNCTION_AUTH_ERROR: 2,
                WEBIM_CONNCTION_OPEN_USERGRID_ERROR: 3,
                WEBIM_CONNCTION_ATTACH_ERROR: 4,
                WEBIM_CONNCTION_ATTACH_USERGRID_ERROR: 5,
                WEBIM_CONNCTION_REOPEN_ERROR: 6,
                WEBIM_CONNCTION_SERVER_CLOSE_ERROR: 7, //7: client-side network offline (net::ERR_INTERNET_DISCONNECTED)
                WEBIM_CONNCTION_SERVER_ERROR: 8, //8: offline by multi login
                WEBIM_CONNCTION_IQ_ERROR: 9,

                WEBIM_CONNCTION_PING_ERROR: 10,
                WEBIM_CONNCTION_NOTIFYVERSION_ERROR: 11,
                WEBIM_CONNCTION_GETROSTER_ERROR: 12,
                WEBIM_CONNCTION_CROSSDOMAIN_ERROR: 13,
                WEBIM_CONNCTION_LISTENING_OUTOF_MAXRETRIES: 14,
                WEBIM_CONNCTION_RECEIVEMSG_CONTENTERROR: 15,
                WEBIM_CONNCTION_DISCONNECTED: 16, //16: server-side close the websocket connection
                WEBIM_CONNCTION_AJAX_ERROR: 17,
                WEBIM_CONNCTION_JOINROOM_ERROR: 18,
                WEBIM_CONNCTION_GETROOM_ERROR: 19,

                WEBIM_CONNCTION_GETROOMINFO_ERROR: 20,
                WEBIM_CONNCTION_GETROOMMEMBER_ERROR: 21,
                WEBIM_CONNCTION_GETROOMOCCUPANTS_ERROR: 22,
                WEBIM_CONNCTION_LOAD_CHATROOM_ERROR: 23,
                WEBIM_CONNCTION_NOT_SUPPORT_CHATROOM_ERROR: 24,
                WEBIM_CONNCTION_JOINCHATROOM_ERROR: 25,
                WEBIM_CONNCTION_QUITCHATROOM_ERROR: 26,
                WEBIM_CONNCTION_APPKEY_NOT_ASSIGN_ERROR: 27,
                WEBIM_CONNCTION_TOKEN_NOT_ASSIGN_ERROR: 28,
                WEBIM_CONNCTION_SESSIONID_NOT_ASSIGN_ERROR: 29,

                WEBIM_CONNCTION_RID_NOT_ASSIGN_ERROR: 30,
                WEBIM_CONNCTION_CALLBACK_INNER_ERROR: 31, //31: 处理下行消息出错 try/catch抛出异常
                WEBIM_CONNCTION_CLIENT_OFFLINE: 32, //32: client offline
                WEBIM_CONNCTION_CLIENT_LOGOUT: 33, //33: client logout
                WEBIM_CONNCTION_CLIENT_TOO_MUCH_ERROR: 34, // 34: Over amount of the tabs a user opened in the same browser
                WEBIM_CONNECTION_ACCEPT_INVITATION_FROM_GROUP: 35,
                WEBIM_CONNECTION_DECLINE_INVITATION_FROM_GROUP: 36,
                WEBIM_CONNECTION_ACCEPT_JOIN_GROUP: 37,
                WEBIM_CONNECTION_DECLINE_JOIN_GROUP: 38,
                WEBIM_CONNECTION_CLOSED: 39,

                WEBIM_UPLOADFILE_BROWSER_ERROR: 100,
                WEBIM_UPLOADFILE_ERROR: 101,
                WEBIM_UPLOADFILE_NO_LOGIN: 102,
                WEBIM_UPLOADFILE_NO_FILE: 103,

                WEBIM_DOWNLOADFILE_ERROR: 200,
                WEBIM_DOWNLOADFILE_NO_LOGIN: 201,
                WEBIM_DOWNLOADFILE_BROWSER_ERROR: 202,

                WEBIM_MESSAGE_REC_TEXT: 300,
                WEBIM_MESSAGE_REC_TEXT_ERROR: 301,
                WEBIM_MESSAGE_REC_EMOTION: 302,
                WEBIM_MESSAGE_REC_PHOTO: 303,
                WEBIM_MESSAGE_REC_AUDIO: 304,
                WEBIM_MESSAGE_REC_AUDIO_FILE: 305,
                WEBIM_MESSAGE_REC_VEDIO: 306,
                WEBIM_MESSAGE_REC_VEDIO_FILE: 307,
                WEBIM_MESSAGE_REC_FILE: 308,
                WEBIM_MESSAGE_SED_TEXT: 309,
                WEBIM_MESSAGE_SED_EMOTION: 310,
                WEBIM_MESSAGE_SED_PHOTO: 311,
                WEBIM_MESSAGE_SED_AUDIO: 312,
                WEBIM_MESSAGE_SED_AUDIO_FILE: 313,
                WEBIM_MESSAGE_SED_VEDIO: 314,
                WEBIM_MESSAGE_SED_VEDIO_FILE: 315,
                WEBIM_MESSAGE_SED_FILE: 316,
                WEBIM_MESSAGE_SED_ERROR: 317,

                STATUS_INIT: 400,
                STATUS_DOLOGIN_USERGRID: 401,
                STATUS_DOLOGIN_IM: 402,
                STATUS_OPENED: 403,
                STATUS_CLOSING: 404,
                STATUS_CLOSED: 405,
                STATUS_ERROR: 406
            };
        })();

        /***/
    },

    /***/ 211:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(213), __webpack_require__(214), __webpack_require__(215), __webpack_require__(216), __webpack_require__(217), __webpack_require__(218), __webpack_require__(219), __webpack_require__(220), __webpack_require__(221), __webpack_require__(222), __webpack_require__(223), __webpack_require__(224), __webpack_require__(225), __webpack_require__(226), __webpack_require__(227), __webpack_require__(228), __webpack_require__(229), __webpack_require__(230), __webpack_require__(231), __webpack_require__(232), __webpack_require__(233), __webpack_require__(234), __webpack_require__(235), __webpack_require__(236), __webpack_require__(237), __webpack_require__(238), __webpack_require__(239), __webpack_require__(240), __webpack_require__(241), __webpack_require__(242), __webpack_require__(243), __webpack_require__(244));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./x64-core", "./lib-typedarrays", "./enc-utf16", "./enc-base64", "./md5", "./sha1", "./sha256", "./sha224", "./sha512", "./sha384", "./sha3", "./ripemd160", "./hmac", "./pbkdf2", "./evpkdf", "./cipher-core", "./mode-cfb", "./mode-ctr", "./mode-ctr-gladman", "./mode-ofb", "./mode-ecb", "./pad-ansix923", "./pad-iso10126", "./pad-iso97971", "./pad-zeropadding", "./pad-nopadding", "./format-hex", "./aes", "./tripledes", "./rc4", "./rabbit", "./rabbit-legacy"], factory);
            }
            else {
                // Global (browser)
                root.CryptoJS = factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            return CryptoJS;

        }));

        /***/
    },

    /***/ 212:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory();
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define([], factory);
            }
            else {
                // Global (browser)
                root.CryptoJS = factory();
            }
        }(this, function () {

            /**
             * CryptoJS core components.
             */
            var CryptoJS = CryptoJS || (function (Math, undefined) {
                /*
		     * Local polyfil of Object.create
		     */
                var create = Object.create || (function () {
                    function F() {
                    };

                    return function (obj) {
                        var subtype;

                        F.prototype = obj;

                        subtype = new F();

                        F.prototype = null;

                        return subtype;
                    };
                }())

                /**
                 * CryptoJS namespace.
                 */
                var C = {};

                /**
                 * Library namespace.
                 */
                var C_lib = C.lib = {};

                /**
                 * Base object for prototypal inheritance.
                 */
                var Base = C_lib.Base = (function () {


                    return {
                        /**
                         * Creates a new object that inherits from this object.
                         *
                         * @param {Object} overrides Properties to copy into the new object.
                         *
                         * @return {Object} The new object.
                         *
                         * @static
                         *
                         * @example
                         *
                         *     var MyType = CryptoJS.lib.Base.extend({
                         *         field: 'value',
                         *
                         *         method: function () {
                         *         }
                         *     });
                         */
                        extend: function (overrides) {
                            // Spawn
                            var subtype = create(this);

                            // Augment
                            if (overrides) {
                                subtype.mixIn(overrides);
                            }

                            // Create default initializer
                            if (!subtype.hasOwnProperty('init') || this.init === subtype.init) {
                                subtype.init = function () {
                                    subtype.$super.init.apply(this, arguments);
                                };
                            }

                            // Initializer's prototype is the subtype object
                            subtype.init.prototype = subtype;

                            // Reference supertype
                            subtype.$super = this;

                            return subtype;
                        },

                        /**
                         * Extends this object and runs the init method.
                         * Arguments to create() will be passed to init().
                         *
                         * @return {Object} The new object.
                         *
                         * @static
                         *
                         * @example
                         *
                         *     var instance = MyType.create();
                         */
                        create: function () {
                            var instance = this.extend();
                            instance.init.apply(instance, arguments);

                            return instance;
                        },

                        /**
                         * Initializes a newly created object.
                         * Override this method to add some logic when your objects are created.
                         *
                         * @example
                         *
                         *     var MyType = CryptoJS.lib.Base.extend({
                         *         init: function () {
                         *             // ...
                         *         }
                         *     });
                         */
                        init: function () {
                        },

                        /**
                         * Copies properties into this object.
                         *
                         * @param {Object} properties The properties to mix in.
                         *
                         * @example
                         *
                         *     MyType.mixIn({
                         *         field: 'value'
                         *     });
                         */
                        mixIn: function (properties) {
                            for (var propertyName in properties) {
                                if (properties.hasOwnProperty(propertyName)) {
                                    this[propertyName] = properties[propertyName];
                                }
                            }

                            // IE won't copy toString using the loop above
                            if (properties.hasOwnProperty('toString')) {
                                this.toString = properties.toString;
                            }
                        },

                        /**
                         * Creates a copy of this object.
                         *
                         * @return {Object} The clone.
                         *
                         * @example
                         *
                         *     var clone = instance.clone();
                         */
                        clone: function () {
                            return this.init.prototype.extend(this);
                        }
                    };
                }());

                /**
                 * An array of 32-bit words.
                 *
                 * @property {Array} words The array of 32-bit words.
                 * @property {number} sigBytes The number of significant bytes in this word array.
                 */
                var WordArray = C_lib.WordArray = Base.extend({
                    /**
                     * Initializes a newly created word array.
                     *
                     * @param {Array} words (Optional) An array of 32-bit words.
                     * @param {number} sigBytes (Optional) The number of significant bytes in the words.
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.lib.WordArray.create();
                     *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);
                     *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);
                     */
                    init: function (words, sigBytes) {
                        words = this.words = words || [];

                        if (sigBytes != undefined) {
                            this.sigBytes = sigBytes;
                        } else {
                            this.sigBytes = words.length * 4;
                        }
                    },

                    /**
                     * Converts this word array to a string.
                     *
                     * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex
                     *
                     * @return {string} The stringified word array.
                     *
                     * @example
                     *
                     *     var string = wordArray + '';
                     *     var string = wordArray.toString();
                     *     var string = wordArray.toString(CryptoJS.enc.Utf8);
                     */
                    toString: function (encoder) {
                        return (encoder || Hex).stringify(this);
                    },

                    /**
                     * Concatenates a word array to this word array.
                     *
                     * @param {WordArray} wordArray The word array to append.
                     *
                     * @return {WordArray} This word array.
                     *
                     * @example
                     *
                     *     wordArray1.concat(wordArray2);
                     */
                    concat: function (wordArray) {
                        // Shortcuts
                        var thisWords = this.words;
                        var thatWords = wordArray.words;
                        var thisSigBytes = this.sigBytes;
                        var thatSigBytes = wordArray.sigBytes;

                        // Clamp excess bits
                        this.clamp();

                        // Concat
                        if (thisSigBytes % 4) {
                            // Copy one byte at a time
                            for (var i = 0; i < thatSigBytes; i++) {
                                var thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
                                thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);
                            }
                        } else {
                            // Copy one word at a time
                            for (var i = 0; i < thatSigBytes; i += 4) {
                                thisWords[(thisSigBytes + i) >>> 2] = thatWords[i >>> 2];
                            }
                        }
                        this.sigBytes += thatSigBytes;

                        // Chainable
                        return this;
                    },

                    /**
                     * Removes insignificant bits.
                     *
                     * @example
                     *
                     *     wordArray.clamp();
                     */
                    clamp: function () {
                        // Shortcuts
                        var words = this.words;
                        var sigBytes = this.sigBytes;

                        // Clamp
                        words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);
                        words.length = Math.ceil(sigBytes / 4);
                    },

                    /**
                     * Creates a copy of this word array.
                     *
                     * @return {WordArray} The clone.
                     *
                     * @example
                     *
                     *     var clone = wordArray.clone();
                     */
                    clone: function () {
                        var clone = Base.clone.call(this);
                        clone.words = this.words.slice(0);

                        return clone;
                    },

                    /**
                     * Creates a word array filled with random bytes.
                     *
                     * @param {number} nBytes The number of random bytes to generate.
                     *
                     * @return {WordArray} The random word array.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.lib.WordArray.random(16);
                     */
                    random: function (nBytes) {
                        var words = [];

                        var r = (function (m_w) {
                            var m_w = m_w;
                            var m_z = 0x3ade68b1;
                            var mask = 0xffffffff;

                            return function () {
                                m_z = (0x9069 * (m_z & 0xFFFF) + (m_z >> 0x10)) & mask;
                                m_w = (0x4650 * (m_w & 0xFFFF) + (m_w >> 0x10)) & mask;
                                var result = ((m_z << 0x10) + m_w) & mask;
                                result /= 0x100000000;
                                result += 0.5;
                                return result * (Math.random() > .5 ? 1 : -1);
                            }
                        });

                        for (var i = 0, rcache; i < nBytes; i += 4) {
                            var _r = r((rcache || Math.random()) * 0x100000000);

                            rcache = _r() * 0x3ade67b7;
                            words.push((_r() * 0x100000000) | 0);
                        }

                        return new WordArray.init(words, nBytes);
                    }
                });

                /**
                 * Encoder namespace.
                 */
                var C_enc = C.enc = {};

                /**
                 * Hex encoding strategy.
                 */
                var Hex = C_enc.Hex = {
                    /**
                     * Converts a word array to a hex string.
                     *
                     * @param {WordArray} wordArray The word array.
                     *
                     * @return {string} The hex string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var hexString = CryptoJS.enc.Hex.stringify(wordArray);
                     */
                    stringify: function (wordArray) {
                        // Shortcuts
                        var words = wordArray.words;
                        var sigBytes = wordArray.sigBytes;

                        // Convert
                        var hexChars = [];
                        for (var i = 0; i < sigBytes; i++) {
                            var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
                            hexChars.push((bite >>> 4).toString(16));
                            hexChars.push((bite & 0x0f).toString(16));
                        }

                        return hexChars.join('');
                    },

                    /**
                     * Converts a hex string to a word array.
                     *
                     * @param {string} hexStr The hex string.
                     *
                     * @return {WordArray} The word array.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.enc.Hex.parse(hexString);
                     */
                    parse: function (hexStr) {
                        // Shortcut
                        var hexStrLength = hexStr.length;

                        // Convert
                        var words = [];
                        for (var i = 0; i < hexStrLength; i += 2) {
                            words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);
                        }

                        return new WordArray.init(words, hexStrLength / 2);
                    }
                };

                /**
                 * Latin1 encoding strategy.
                 */
                var Latin1 = C_enc.Latin1 = {
                    /**
                     * Converts a word array to a Latin1 string.
                     *
                     * @param {WordArray} wordArray The word array.
                     *
                     * @return {string} The Latin1 string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);
                     */
                    stringify: function (wordArray) {
                        // Shortcuts
                        var words = wordArray.words;
                        var sigBytes = wordArray.sigBytes;

                        // Convert
                        var latin1Chars = [];
                        for (var i = 0; i < sigBytes; i++) {
                            var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
                            latin1Chars.push(String.fromCharCode(bite));
                        }

                        return latin1Chars.join('');
                    },

                    /**
                     * Converts a Latin1 string to a word array.
                     *
                     * @param {string} latin1Str The Latin1 string.
                     *
                     * @return {WordArray} The word array.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.enc.Latin1.parse(latin1String);
                     */
                    parse: function (latin1Str) {
                        // Shortcut
                        var latin1StrLength = latin1Str.length;

                        // Convert
                        var words = [];
                        for (var i = 0; i < latin1StrLength; i++) {
                            words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);
                        }

                        return new WordArray.init(words, latin1StrLength);
                    }
                };

                /**
                 * UTF-8 encoding strategy.
                 */
                var Utf8 = C_enc.Utf8 = {
                    /**
                     * Converts a word array to a UTF-8 string.
                     *
                     * @param {WordArray} wordArray The word array.
                     *
                     * @return {string} The UTF-8 string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);
                     */
                    stringify: function (wordArray) {
                        try {
                            return decodeURIComponent(escape(Latin1.stringify(wordArray)));
                        } catch (e) {
                            throw new Error('Malformed UTF-8 data');
                        }
                    },

                    /**
                     * Converts a UTF-8 string to a word array.
                     *
                     * @param {string} utf8Str The UTF-8 string.
                     *
                     * @return {WordArray} The word array.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.enc.Utf8.parse(utf8String);
                     */
                    parse: function (utf8Str) {
                        return Latin1.parse(unescape(encodeURIComponent(utf8Str)));
                    }
                };

                /**
                 * Abstract buffered block algorithm template.
                 *
                 * The property blockSize must be implemented in a concrete subtype.
                 *
                 * @property {number} _minBufferSize The number of blocks that should be kept unprocessed in the buffer. Default: 0
                 */
                var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm = Base.extend({
                    /**
                     * Resets this block algorithm's data buffer to its initial state.
                     *
                     * @example
                     *
                     *     bufferedBlockAlgorithm.reset();
                     */
                    reset: function () {
                        // Initial values
                        this._data = new WordArray.init();
                        this._nDataBytes = 0;
                    },

                    /**
                     * Adds new data to this block algorithm's buffer.
                     *
                     * @param {WordArray|string} data The data to append. Strings are converted to a WordArray using UTF-8.
                     *
                     * @example
                     *
                     *     bufferedBlockAlgorithm._append('data');
                     *     bufferedBlockAlgorithm._append(wordArray);
                     */
                    _append: function (data) {
                        // Convert string to WordArray, else assume WordArray already
                        if (typeof data == 'string') {
                            data = Utf8.parse(data);
                        }

                        // Append
                        this._data.concat(data);
                        this._nDataBytes += data.sigBytes;
                    },

                    /**
                     * Processes available data blocks.
                     *
                     * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.
                     *
                     * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.
                     *
                     * @return {WordArray} The processed data.
                     *
                     * @example
                     *
                     *     var processedData = bufferedBlockAlgorithm._process();
                     *     var processedData = bufferedBlockAlgorithm._process(!!'flush');
                     */
                    _process: function (doFlush) {
                        // Shortcuts
                        var data = this._data;
                        var dataWords = data.words;
                        var dataSigBytes = data.sigBytes;
                        var blockSize = this.blockSize;
                        var blockSizeBytes = blockSize * 4;

                        // Count blocks ready
                        var nBlocksReady = dataSigBytes / blockSizeBytes;
                        if (doFlush) {
                            // Round up to include partial blocks
                            nBlocksReady = Math.ceil(nBlocksReady);
                        } else {
                            // Round down to include only full blocks,
                            // less the number of blocks that must remain in the buffer
                            nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);
                        }

                        // Count words ready
                        var nWordsReady = nBlocksReady * blockSize;

                        // Count bytes ready
                        var nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);

                        // Process blocks
                        if (nWordsReady) {
                            for (var offset = 0; offset < nWordsReady; offset += blockSize) {
                                // Perform concrete-algorithm logic
                                this._doProcessBlock(dataWords, offset);
                            }

                            // Remove processed words
                            var processedWords = dataWords.splice(0, nWordsReady);
                            data.sigBytes -= nBytesReady;
                        }

                        // Return processed words
                        return new WordArray.init(processedWords, nBytesReady);
                    },

                    /**
                     * Creates a copy of this object.
                     *
                     * @return {Object} The clone.
                     *
                     * @example
                     *
                     *     var clone = bufferedBlockAlgorithm.clone();
                     */
                    clone: function () {
                        var clone = Base.clone.call(this);
                        clone._data = this._data.clone();

                        return clone;
                    },

                    _minBufferSize: 0
                });

                /**
                 * Abstract hasher template.
                 *
                 * @property {number} blockSize The number of 32-bit words this hasher operates on. Default: 16 (512 bits)
                 */
                var Hasher = C_lib.Hasher = BufferedBlockAlgorithm.extend({
                    /**
                     * Configuration options.
                     */
                    cfg: Base.extend(),

                    /**
                     * Initializes a newly created hasher.
                     *
                     * @param {Object} cfg (Optional) The configuration options to use for this hash computation.
                     *
                     * @example
                     *
                     *     var hasher = CryptoJS.algo.SHA256.create();
                     */
                    init: function (cfg) {
                        // Apply config defaults
                        this.cfg = this.cfg.extend(cfg);

                        // Set initial values
                        this.reset();
                    },

                    /**
                     * Resets this hasher to its initial state.
                     *
                     * @example
                     *
                     *     hasher.reset();
                     */
                    reset: function () {
                        // Reset data buffer
                        BufferedBlockAlgorithm.reset.call(this);

                        // Perform concrete-hasher logic
                        this._doReset();
                    },

                    /**
                     * Updates this hasher with a message.
                     *
                     * @param {WordArray|string} messageUpdate The message to append.
                     *
                     * @return {Hasher} This hasher.
                     *
                     * @example
                     *
                     *     hasher.update('message');
                     *     hasher.update(wordArray);
                     */
                    update: function (messageUpdate) {
                        // Append
                        this._append(messageUpdate);

                        // Update the hash
                        this._process();

                        // Chainable
                        return this;
                    },

                    /**
                     * Finalizes the hash computation.
                     * Note that the finalize operation is effectively a destructive, read-once operation.
                     *
                     * @param {WordArray|string} messageUpdate (Optional) A final message update.
                     *
                     * @return {WordArray} The hash.
                     *
                     * @example
                     *
                     *     var hash = hasher.finalize();
                     *     var hash = hasher.finalize('message');
                     *     var hash = hasher.finalize(wordArray);
                     */
                    finalize: function (messageUpdate) {
                        // Final message update
                        if (messageUpdate) {
                            this._append(messageUpdate);
                        }

                        // Perform concrete-hasher logic
                        var hash = this._doFinalize();

                        return hash;
                    },

                    blockSize: 512 / 32,

                    /**
                     * Creates a shortcut function to a hasher's object interface.
                     *
                     * @param {Hasher} hasher The hasher to create a helper for.
                     *
                     * @return {Function} The shortcut function.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);
                     */
                    _createHelper: function (hasher) {
                        return function (message, cfg) {
                            return new hasher.init(cfg).finalize(message);
                        };
                    },

                    /**
                     * Creates a shortcut function to the HMAC's object interface.
                     *
                     * @param {Hasher} hasher The hasher to use in this HMAC helper.
                     *
                     * @return {Function} The shortcut function.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);
                     */
                    _createHmacHelper: function (hasher) {
                        return function (message, key) {
                            return new C_algo.HMAC.init(hasher, key).finalize(message);
                        };
                    }
                });

                /**
                 * Algorithm namespace.
                 */
                var C_algo = C.algo = {};

                return C;
            }(Math));


            return CryptoJS;

        }));

        /***/
    },

    /***/ 213:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function (undefined) {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var Base = C_lib.Base;
                var X32WordArray = C_lib.WordArray;

                /**
                 * x64 namespace.
                 */
                var C_x64 = C.x64 = {};

                /**
                 * A 64-bit word.
                 */
                var X64Word = C_x64.Word = Base.extend({
                    /**
                     * Initializes a newly created 64-bit word.
                     *
                     * @param {number} high The high 32 bits.
                     * @param {number} low The low 32 bits.
                     *
                     * @example
                     *
                     *     var x64Word = CryptoJS.x64.Word.create(0x00010203, 0x04050607);
                     */
                    init: function (high, low) {
                        this.high = high;
                        this.low = low;
                    }

                    /**
                     * Bitwise NOTs this word.
                     *
                     * @return {X64Word} A new x64-Word object after negating.
                     *
                     * @example
                     *
                     *     var negated = x64Word.not();
                     */
                    // not: function () {
                    // var high = ~this.high;
                    // var low = ~this.low;

                    // return X64Word.create(high, low);
                    // },

                    /**
                     * Bitwise ANDs this word with the passed word.
                     *
                     * @param {X64Word} word The x64-Word to AND with this word.
                     *
                     * @return {X64Word} A new x64-Word object after ANDing.
                     *
                     * @example
                     *
                     *     var anded = x64Word.and(anotherX64Word);
                     */
                    // and: function (word) {
                    // var high = this.high & word.high;
                    // var low = this.low & word.low;

                    // return X64Word.create(high, low);
                    // },

                    /**
                     * Bitwise ORs this word with the passed word.
                     *
                     * @param {X64Word} word The x64-Word to OR with this word.
                     *
                     * @return {X64Word} A new x64-Word object after ORing.
                     *
                     * @example
                     *
                     *     var ored = x64Word.or(anotherX64Word);
                     */
                    // or: function (word) {
                    // var high = this.high | word.high;
                    // var low = this.low | word.low;

                    // return X64Word.create(high, low);
                    // },

                    /**
                     * Bitwise XORs this word with the passed word.
                     *
                     * @param {X64Word} word The x64-Word to XOR with this word.
                     *
                     * @return {X64Word} A new x64-Word object after XORing.
                     *
                     * @example
                     *
                     *     var xored = x64Word.xor(anotherX64Word);
                     */
                    // xor: function (word) {
                    // var high = this.high ^ word.high;
                    // var low = this.low ^ word.low;

                    // return X64Word.create(high, low);
                    // },

                    /**
                     * Shifts this word n bits to the left.
                     *
                     * @param {number} n The number of bits to shift.
                     *
                     * @return {X64Word} A new x64-Word object after shifting.
                     *
                     * @example
                     *
                     *     var shifted = x64Word.shiftL(25);
                     */
                    // shiftL: function (n) {
                    // if (n < 32) {
                    // var high = (this.high << n) | (this.low >>> (32 - n));
                    // var low = this.low << n;
                    // } else {
                    // var high = this.low << (n - 32);
                    // var low = 0;
                    // }

                    // return X64Word.create(high, low);
                    // },

                    /**
                     * Shifts this word n bits to the right.
                     *
                     * @param {number} n The number of bits to shift.
                     *
                     * @return {X64Word} A new x64-Word object after shifting.
                     *
                     * @example
                     *
                     *     var shifted = x64Word.shiftR(7);
                     */
                    // shiftR: function (n) {
                    // if (n < 32) {
                    // var low = (this.low >>> n) | (this.high << (32 - n));
                    // var high = this.high >>> n;
                    // } else {
                    // var low = this.high >>> (n - 32);
                    // var high = 0;
                    // }

                    // return X64Word.create(high, low);
                    // },

                    /**
                     * Rotates this word n bits to the left.
                     *
                     * @param {number} n The number of bits to rotate.
                     *
                     * @return {X64Word} A new x64-Word object after rotating.
                     *
                     * @example
                     *
                     *     var rotated = x64Word.rotL(25);
                     */
                    // rotL: function (n) {
                    // return this.shiftL(n).or(this.shiftR(64 - n));
                    // },

                    /**
                     * Rotates this word n bits to the right.
                     *
                     * @param {number} n The number of bits to rotate.
                     *
                     * @return {X64Word} A new x64-Word object after rotating.
                     *
                     * @example
                     *
                     *     var rotated = x64Word.rotR(7);
                     */
                    // rotR: function (n) {
                    // return this.shiftR(n).or(this.shiftL(64 - n));
                    // },

                    /**
                     * Adds this word with the passed word.
                     *
                     * @param {X64Word} word The x64-Word to add with this word.
                     *
                     * @return {X64Word} A new x64-Word object after adding.
                     *
                     * @example
                     *
                     *     var added = x64Word.add(anotherX64Word);
                     */
                    // add: function (word) {
                    // var low = (this.low + word.low) | 0;
                    // var carry = (low >>> 0) < (this.low >>> 0) ? 1 : 0;
                    // var high = (this.high + word.high + carry) | 0;

                    // return X64Word.create(high, low);
                    // }
                });

                /**
                 * An array of 64-bit words.
                 *
                 * @property {Array} words The array of CryptoJS.x64.Word objects.
                 * @property {number} sigBytes The number of significant bytes in this word array.
                 */
                var X64WordArray = C_x64.WordArray = Base.extend({
                    /**
                     * Initializes a newly created word array.
                     *
                     * @param {Array} words (Optional) An array of CryptoJS.x64.Word objects.
                     * @param {number} sigBytes (Optional) The number of significant bytes in the words.
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.x64.WordArray.create();
                     *
                     *     var wordArray = CryptoJS.x64.WordArray.create([
                     *         CryptoJS.x64.Word.create(0x00010203, 0x04050607),
                     *         CryptoJS.x64.Word.create(0x18191a1b, 0x1c1d1e1f)
                     *     ]);
                     *
                     *     var wordArray = CryptoJS.x64.WordArray.create([
                     *         CryptoJS.x64.Word.create(0x00010203, 0x04050607),
                     *         CryptoJS.x64.Word.create(0x18191a1b, 0x1c1d1e1f)
                     *     ], 10);
                     */
                    init: function (words, sigBytes) {
                        words = this.words = words || [];

                        if (sigBytes != undefined) {
                            this.sigBytes = sigBytes;
                        } else {
                            this.sigBytes = words.length * 8;
                        }
                    },

                    /**
                     * Converts this 64-bit word array to a 32-bit word array.
                     *
                     * @return {CryptoJS.lib.WordArray} This word array's data as a 32-bit word array.
                     *
                     * @example
                     *
                     *     var x32WordArray = x64WordArray.toX32();
                     */
                    toX32: function () {
                        // Shortcuts
                        var x64Words = this.words;
                        var x64WordsLength = x64Words.length;

                        // Convert
                        var x32Words = [];
                        for (var i = 0; i < x64WordsLength; i++) {
                            var x64Word = x64Words[i];
                            x32Words.push(x64Word.high);
                            x32Words.push(x64Word.low);
                        }

                        return X32WordArray.create(x32Words, this.sigBytes);
                    },

                    /**
                     * Creates a copy of this word array.
                     *
                     * @return {X64WordArray} The clone.
                     *
                     * @example
                     *
                     *     var clone = x64WordArray.clone();
                     */
                    clone: function () {
                        var clone = Base.clone.call(this);

                        // Clone "words" array
                        var words = clone.words = this.words.slice(0);

                        // Clone each X64Word object
                        var wordsLength = words.length;
                        for (var i = 0; i < wordsLength; i++) {
                            words[i] = words[i].clone();
                        }

                        return clone;
                    }
                });
            }());


            return CryptoJS;

        }));

        /***/
    },

    /***/ 214:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Check if typed arrays are supported
                if (typeof ArrayBuffer != 'function') {
                    return;
                }

                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;

                // Reference original init
                var superInit = WordArray.init;

                // Augment WordArray.init to handle typed arrays
                var subInit = WordArray.init = function (typedArray) {
                    // Convert buffers to uint8
                    if (typedArray instanceof ArrayBuffer) {
                        typedArray = new Uint8Array(typedArray);
                    }

                    // Convert other array views to uint8
                    if (
                        typedArray instanceof Int8Array ||
                        (typeof Uint8ClampedArray !== "undefined" && typedArray instanceof Uint8ClampedArray) ||
                        typedArray instanceof Int16Array ||
                        typedArray instanceof Uint16Array ||
                        typedArray instanceof Int32Array ||
                        typedArray instanceof Uint32Array ||
                        typedArray instanceof Float32Array ||
                        typedArray instanceof Float64Array
                    ) {
                        typedArray = new Uint8Array(typedArray.buffer, typedArray.byteOffset, typedArray.byteLength);
                    }

                    // Handle Uint8Array
                    if (typedArray instanceof Uint8Array) {
                        // Shortcut
                        var typedArrayByteLength = typedArray.byteLength;

                        // Extract bytes
                        var words = [];
                        for (var i = 0; i < typedArrayByteLength; i++) {
                            words[i >>> 2] |= typedArray[i] << (24 - (i % 4) * 8);
                        }

                        // Initialize this word array
                        superInit.call(this, words, typedArrayByteLength);
                    } else {
                        // Else call normal init
                        superInit.apply(this, arguments);
                    }
                };

                subInit.prototype = WordArray;
            }());


            return CryptoJS.lib.WordArray;

        }));

        /***/
    },

    /***/ 215:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var C_enc = C.enc;

                /**
                 * UTF-16 BE encoding strategy.
                 */
                var Utf16BE = C_enc.Utf16 = C_enc.Utf16BE = {
                    /**
                     * Converts a word array to a UTF-16 BE string.
                     *
                     * @param {WordArray} wordArray The word array.
                     *
                     * @return {string} The UTF-16 BE string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var utf16String = CryptoJS.enc.Utf16.stringify(wordArray);
                     */
                    stringify: function (wordArray) {
                        // Shortcuts
                        var words = wordArray.words;
                        var sigBytes = wordArray.sigBytes;

                        // Convert
                        var utf16Chars = [];
                        for (var i = 0; i < sigBytes; i += 2) {
                            var codePoint = (words[i >>> 2] >>> (16 - (i % 4) * 8)) & 0xffff;
                            utf16Chars.push(String.fromCharCode(codePoint));
                        }

                        return utf16Chars.join('');
                    },

                    /**
                     * Converts a UTF-16 BE string to a word array.
                     *
                     * @param {string} utf16Str The UTF-16 BE string.
                     *
                     * @return {WordArray} The word array.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.enc.Utf16.parse(utf16String);
                     */
                    parse: function (utf16Str) {
                        // Shortcut
                        var utf16StrLength = utf16Str.length;

                        // Convert
                        var words = [];
                        for (var i = 0; i < utf16StrLength; i++) {
                            words[i >>> 1] |= utf16Str.charCodeAt(i) << (16 - (i % 2) * 16);
                        }

                        return WordArray.create(words, utf16StrLength * 2);
                    }
                };

                /**
                 * UTF-16 LE encoding strategy.
                 */
                C_enc.Utf16LE = {
                    /**
                     * Converts a word array to a UTF-16 LE string.
                     *
                     * @param {WordArray} wordArray The word array.
                     *
                     * @return {string} The UTF-16 LE string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var utf16Str = CryptoJS.enc.Utf16LE.stringify(wordArray);
                     */
                    stringify: function (wordArray) {
                        // Shortcuts
                        var words = wordArray.words;
                        var sigBytes = wordArray.sigBytes;

                        // Convert
                        var utf16Chars = [];
                        for (var i = 0; i < sigBytes; i += 2) {
                            var codePoint = swapEndian((words[i >>> 2] >>> (16 - (i % 4) * 8)) & 0xffff);
                            utf16Chars.push(String.fromCharCode(codePoint));
                        }

                        return utf16Chars.join('');
                    },

                    /**
                     * Converts a UTF-16 LE string to a word array.
                     *
                     * @param {string} utf16Str The UTF-16 LE string.
                     *
                     * @return {WordArray} The word array.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.enc.Utf16LE.parse(utf16Str);
                     */
                    parse: function (utf16Str) {
                        // Shortcut
                        var utf16StrLength = utf16Str.length;

                        // Convert
                        var words = [];
                        for (var i = 0; i < utf16StrLength; i++) {
                            words[i >>> 1] |= swapEndian(utf16Str.charCodeAt(i) << (16 - (i % 2) * 16));
                        }

                        return WordArray.create(words, utf16StrLength * 2);
                    }
                };

                function swapEndian(word) {
                    return ((word << 8) & 0xff00ff00) | ((word >>> 8) & 0x00ff00ff);
                }
            }());


            return CryptoJS.enc.Utf16;

        }));

        /***/
    },

    /***/ 216:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var C_enc = C.enc;

                /**
                 * Base64 encoding strategy.
                 */
                var Base64 = C_enc.Base64 = {
                    /**
                     * Converts a word array to a Base64 string.
                     *
                     * @param {WordArray} wordArray The word array.
                     *
                     * @return {string} The Base64 string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var base64String = CryptoJS.enc.Base64.stringify(wordArray);
                     */
                    stringify: function (wordArray) {
                        // Shortcuts
                        var words = wordArray.words;
                        var sigBytes = wordArray.sigBytes;
                        var map = this._map;

                        // Clamp excess bits
                        wordArray.clamp();

                        // Convert
                        var base64Chars = [];
                        for (var i = 0; i < sigBytes; i += 3) {
                            var byte1 = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
                            var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;
                            var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;

                            var triplet = (byte1 << 16) | (byte2 << 8) | byte3;

                            for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {
                                base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));
                            }
                        }

                        // Add padding
                        var paddingChar = map.charAt(64);
                        if (paddingChar) {
                            while (base64Chars.length % 4) {
                                base64Chars.push(paddingChar);
                            }
                        }

                        return base64Chars.join('');
                    },

                    /**
                     * Converts a Base64 string to a word array.
                     *
                     * @param {string} base64Str The Base64 string.
                     *
                     * @return {WordArray} The word array.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var wordArray = CryptoJS.enc.Base64.parse(base64String);
                     */
                    parse: function (base64Str) {
                        // Shortcuts
                        var base64StrLength = base64Str.length;
                        var map = this._map;
                        var reverseMap = this._reverseMap;

                        if (!reverseMap) {
                            reverseMap = this._reverseMap = [];
                            for (var j = 0; j < map.length; j++) {
                                reverseMap[map.charCodeAt(j)] = j;
                            }
                        }

                        // Ignore padding
                        var paddingChar = map.charAt(64);
                        if (paddingChar) {
                            var paddingIndex = base64Str.indexOf(paddingChar);
                            if (paddingIndex !== -1) {
                                base64StrLength = paddingIndex;
                            }
                        }

                        // Convert
                        return parseLoop(base64Str, base64StrLength, reverseMap);

                    },

                    _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='
                };

                function parseLoop(base64Str, base64StrLength, reverseMap) {
                    var words = [];
                    var nBytes = 0;
                    for (var i = 0; i < base64StrLength; i++) {
                        if (i % 4) {
                            var bits1 = reverseMap[base64Str.charCodeAt(i - 1)] << ((i % 4) * 2);
                            var bits2 = reverseMap[base64Str.charCodeAt(i)] >>> (6 - (i % 4) * 2);
                            words[nBytes >>> 2] |= (bits1 | bits2) << (24 - (nBytes % 4) * 8);
                            nBytes++;
                        }
                    }
                    return WordArray.create(words, nBytes);
                }
            }());


            return CryptoJS.enc.Base64;

        }));

        /***/
    },

    /***/ 217:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function (Math) {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var Hasher = C_lib.Hasher;
                var C_algo = C.algo;

                // Constants table
                var T = [];

                // Compute constants
                (function () {
                    for (var i = 0; i < 64; i++) {
                        T[i] = (Math.abs(Math.sin(i + 1)) * 0x100000000) | 0;
                    }
                }());

                /**
                 * MD5 hash algorithm.
                 */
                var MD5 = C_algo.MD5 = Hasher.extend({
                    _doReset: function () {
                        this._hash = new WordArray.init([
                            0x67452301, 0xefcdab89,
                            0x98badcfe, 0x10325476
                        ]);
                    },

                    _doProcessBlock: function (M, offset) {
                        // Swap endian
                        for (var i = 0; i < 16; i++) {
                            // Shortcuts
                            var offset_i = offset + i;
                            var M_offset_i = M[offset_i];

                            M[offset_i] = (
                                (((M_offset_i << 8) | (M_offset_i >>> 24)) & 0x00ff00ff) |
                                (((M_offset_i << 24) | (M_offset_i >>> 8)) & 0xff00ff00)
                            );
                        }

                        // Shortcuts
                        var H = this._hash.words;

                        var M_offset_0 = M[offset + 0];
                        var M_offset_1 = M[offset + 1];
                        var M_offset_2 = M[offset + 2];
                        var M_offset_3 = M[offset + 3];
                        var M_offset_4 = M[offset + 4];
                        var M_offset_5 = M[offset + 5];
                        var M_offset_6 = M[offset + 6];
                        var M_offset_7 = M[offset + 7];
                        var M_offset_8 = M[offset + 8];
                        var M_offset_9 = M[offset + 9];
                        var M_offset_10 = M[offset + 10];
                        var M_offset_11 = M[offset + 11];
                        var M_offset_12 = M[offset + 12];
                        var M_offset_13 = M[offset + 13];
                        var M_offset_14 = M[offset + 14];
                        var M_offset_15 = M[offset + 15];

                        // Working varialbes
                        var a = H[0];
                        var b = H[1];
                        var c = H[2];
                        var d = H[3];

                        // Computation
                        a = FF(a, b, c, d, M_offset_0, 7, T[0]);
                        d = FF(d, a, b, c, M_offset_1, 12, T[1]);
                        c = FF(c, d, a, b, M_offset_2, 17, T[2]);
                        b = FF(b, c, d, a, M_offset_3, 22, T[3]);
                        a = FF(a, b, c, d, M_offset_4, 7, T[4]);
                        d = FF(d, a, b, c, M_offset_5, 12, T[5]);
                        c = FF(c, d, a, b, M_offset_6, 17, T[6]);
                        b = FF(b, c, d, a, M_offset_7, 22, T[7]);
                        a = FF(a, b, c, d, M_offset_8, 7, T[8]);
                        d = FF(d, a, b, c, M_offset_9, 12, T[9]);
                        c = FF(c, d, a, b, M_offset_10, 17, T[10]);
                        b = FF(b, c, d, a, M_offset_11, 22, T[11]);
                        a = FF(a, b, c, d, M_offset_12, 7, T[12]);
                        d = FF(d, a, b, c, M_offset_13, 12, T[13]);
                        c = FF(c, d, a, b, M_offset_14, 17, T[14]);
                        b = FF(b, c, d, a, M_offset_15, 22, T[15]);

                        a = GG(a, b, c, d, M_offset_1, 5, T[16]);
                        d = GG(d, a, b, c, M_offset_6, 9, T[17]);
                        c = GG(c, d, a, b, M_offset_11, 14, T[18]);
                        b = GG(b, c, d, a, M_offset_0, 20, T[19]);
                        a = GG(a, b, c, d, M_offset_5, 5, T[20]);
                        d = GG(d, a, b, c, M_offset_10, 9, T[21]);
                        c = GG(c, d, a, b, M_offset_15, 14, T[22]);
                        b = GG(b, c, d, a, M_offset_4, 20, T[23]);
                        a = GG(a, b, c, d, M_offset_9, 5, T[24]);
                        d = GG(d, a, b, c, M_offset_14, 9, T[25]);
                        c = GG(c, d, a, b, M_offset_3, 14, T[26]);
                        b = GG(b, c, d, a, M_offset_8, 20, T[27]);
                        a = GG(a, b, c, d, M_offset_13, 5, T[28]);
                        d = GG(d, a, b, c, M_offset_2, 9, T[29]);
                        c = GG(c, d, a, b, M_offset_7, 14, T[30]);
                        b = GG(b, c, d, a, M_offset_12, 20, T[31]);

                        a = HH(a, b, c, d, M_offset_5, 4, T[32]);
                        d = HH(d, a, b, c, M_offset_8, 11, T[33]);
                        c = HH(c, d, a, b, M_offset_11, 16, T[34]);
                        b = HH(b, c, d, a, M_offset_14, 23, T[35]);
                        a = HH(a, b, c, d, M_offset_1, 4, T[36]);
                        d = HH(d, a, b, c, M_offset_4, 11, T[37]);
                        c = HH(c, d, a, b, M_offset_7, 16, T[38]);
                        b = HH(b, c, d, a, M_offset_10, 23, T[39]);
                        a = HH(a, b, c, d, M_offset_13, 4, T[40]);
                        d = HH(d, a, b, c, M_offset_0, 11, T[41]);
                        c = HH(c, d, a, b, M_offset_3, 16, T[42]);
                        b = HH(b, c, d, a, M_offset_6, 23, T[43]);
                        a = HH(a, b, c, d, M_offset_9, 4, T[44]);
                        d = HH(d, a, b, c, M_offset_12, 11, T[45]);
                        c = HH(c, d, a, b, M_offset_15, 16, T[46]);
                        b = HH(b, c, d, a, M_offset_2, 23, T[47]);

                        a = II(a, b, c, d, M_offset_0, 6, T[48]);
                        d = II(d, a, b, c, M_offset_7, 10, T[49]);
                        c = II(c, d, a, b, M_offset_14, 15, T[50]);
                        b = II(b, c, d, a, M_offset_5, 21, T[51]);
                        a = II(a, b, c, d, M_offset_12, 6, T[52]);
                        d = II(d, a, b, c, M_offset_3, 10, T[53]);
                        c = II(c, d, a, b, M_offset_10, 15, T[54]);
                        b = II(b, c, d, a, M_offset_1, 21, T[55]);
                        a = II(a, b, c, d, M_offset_8, 6, T[56]);
                        d = II(d, a, b, c, M_offset_15, 10, T[57]);
                        c = II(c, d, a, b, M_offset_6, 15, T[58]);
                        b = II(b, c, d, a, M_offset_13, 21, T[59]);
                        a = II(a, b, c, d, M_offset_4, 6, T[60]);
                        d = II(d, a, b, c, M_offset_11, 10, T[61]);
                        c = II(c, d, a, b, M_offset_2, 15, T[62]);
                        b = II(b, c, d, a, M_offset_9, 21, T[63]);

                        // Intermediate hash value
                        H[0] = (H[0] + a) | 0;
                        H[1] = (H[1] + b) | 0;
                        H[2] = (H[2] + c) | 0;
                        H[3] = (H[3] + d) | 0;
                    },

                    _doFinalize: function () {
                        // Shortcuts
                        var data = this._data;
                        var dataWords = data.words;

                        var nBitsTotal = this._nDataBytes * 8;
                        var nBitsLeft = data.sigBytes * 8;

                        // Add padding
                        dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);

                        var nBitsTotalH = Math.floor(nBitsTotal / 0x100000000);
                        var nBitsTotalL = nBitsTotal;
                        dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = (
                            (((nBitsTotalH << 8) | (nBitsTotalH >>> 24)) & 0x00ff00ff) |
                            (((nBitsTotalH << 24) | (nBitsTotalH >>> 8)) & 0xff00ff00)
                        );
                        dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = (
                            (((nBitsTotalL << 8) | (nBitsTotalL >>> 24)) & 0x00ff00ff) |
                            (((nBitsTotalL << 24) | (nBitsTotalL >>> 8)) & 0xff00ff00)
                        );

                        data.sigBytes = (dataWords.length + 1) * 4;

                        // Hash final blocks
                        this._process();

                        // Shortcuts
                        var hash = this._hash;
                        var H = hash.words;

                        // Swap endian
                        for (var i = 0; i < 4; i++) {
                            // Shortcut
                            var H_i = H[i];

                            H[i] = (((H_i << 8) | (H_i >>> 24)) & 0x00ff00ff) |
                                (((H_i << 24) | (H_i >>> 8)) & 0xff00ff00);
                        }

                        // Return final computed hash
                        return hash;
                    },

                    clone: function () {
                        var clone = Hasher.clone.call(this);
                        clone._hash = this._hash.clone();

                        return clone;
                    }
                });

                function FF(a, b, c, d, x, s, t) {
                    var n = a + ((b & c) | (~b & d)) + x + t;
                    return ((n << s) | (n >>> (32 - s))) + b;
                }

                function GG(a, b, c, d, x, s, t) {
                    var n = a + ((b & d) | (c & ~d)) + x + t;
                    return ((n << s) | (n >>> (32 - s))) + b;
                }

                function HH(a, b, c, d, x, s, t) {
                    var n = a + (b ^ c ^ d) + x + t;
                    return ((n << s) | (n >>> (32 - s))) + b;
                }

                function II(a, b, c, d, x, s, t) {
                    var n = a + (c ^ (b | ~d)) + x + t;
                    return ((n << s) | (n >>> (32 - s))) + b;
                }

                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.MD5('message');
                 *     var hash = CryptoJS.MD5(wordArray);
                 */
                C.MD5 = Hasher._createHelper(MD5);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacMD5(message, key);
                 */
                C.HmacMD5 = Hasher._createHmacHelper(MD5);
            }(Math));


            return CryptoJS.MD5;

        }));

        /***/
    },

    /***/ 218:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var Hasher = C_lib.Hasher;
                var C_algo = C.algo;

                // Reusable object
                var W = [];

                /**
                 * SHA-1 hash algorithm.
                 */
                var SHA1 = C_algo.SHA1 = Hasher.extend({
                    _doReset: function () {
                        this._hash = new WordArray.init([
                            0x67452301, 0xefcdab89,
                            0x98badcfe, 0x10325476,
                            0xc3d2e1f0
                        ]);
                    },

                    _doProcessBlock: function (M, offset) {
                        // Shortcut
                        var H = this._hash.words;

                        // Working variables
                        var a = H[0];
                        var b = H[1];
                        var c = H[2];
                        var d = H[3];
                        var e = H[4];

                        // Computation
                        for (var i = 0; i < 80; i++) {
                            if (i < 16) {
                                W[i] = M[offset + i] | 0;
                            } else {
                                var n = W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16];
                                W[i] = (n << 1) | (n >>> 31);
                            }

                            var t = ((a << 5) | (a >>> 27)) + e + W[i];
                            if (i < 20) {
                                t += ((b & c) | (~b & d)) + 0x5a827999;
                            } else if (i < 40) {
                                t += (b ^ c ^ d) + 0x6ed9eba1;
                            } else if (i < 60) {
                                t += ((b & c) | (b & d) | (c & d)) - 0x70e44324;
                            } else /* if (i < 80) */ {
                                t += (b ^ c ^ d) - 0x359d3e2a;
                            }

                            e = d;
                            d = c;
                            c = (b << 30) | (b >>> 2);
                            b = a;
                            a = t;
                        }

                        // Intermediate hash value
                        H[0] = (H[0] + a) | 0;
                        H[1] = (H[1] + b) | 0;
                        H[2] = (H[2] + c) | 0;
                        H[3] = (H[3] + d) | 0;
                        H[4] = (H[4] + e) | 0;
                    },

                    _doFinalize: function () {
                        // Shortcuts
                        var data = this._data;
                        var dataWords = data.words;

                        var nBitsTotal = this._nDataBytes * 8;
                        var nBitsLeft = data.sigBytes * 8;

                        // Add padding
                        dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
                        dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
                        dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
                        data.sigBytes = dataWords.length * 4;

                        // Hash final blocks
                        this._process();

                        // Return final computed hash
                        return this._hash;
                    },

                    clone: function () {
                        var clone = Hasher.clone.call(this);
                        clone._hash = this._hash.clone();

                        return clone;
                    }
                });

                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.SHA1('message');
                 *     var hash = CryptoJS.SHA1(wordArray);
                 */
                C.SHA1 = Hasher._createHelper(SHA1);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacSHA1(message, key);
                 */
                C.HmacSHA1 = Hasher._createHmacHelper(SHA1);
            }());


            return CryptoJS.SHA1;

        }));

        /***/
    },

    /***/ 219:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function (Math) {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var Hasher = C_lib.Hasher;
                var C_algo = C.algo;

                // Initialization and round constants tables
                var H = [];
                var K = [];

                // Compute constants
                (function () {
                    function isPrime(n) {
                        var sqrtN = Math.sqrt(n);
                        for (var factor = 2; factor <= sqrtN; factor++) {
                            if (!(n % factor)) {
                                return false;
                            }
                        }

                        return true;
                    }

                    function getFractionalBits(n) {
                        return ((n - (n | 0)) * 0x100000000) | 0;
                    }

                    var n = 2;
                    var nPrime = 0;
                    while (nPrime < 64) {
                        if (isPrime(n)) {
                            if (nPrime < 8) {
                                H[nPrime] = getFractionalBits(Math.pow(n, 1 / 2));
                            }
                            K[nPrime] = getFractionalBits(Math.pow(n, 1 / 3));

                            nPrime++;
                        }

                        n++;
                    }
                }());

                // Reusable object
                var W = [];

                /**
                 * SHA-256 hash algorithm.
                 */
                var SHA256 = C_algo.SHA256 = Hasher.extend({
                    _doReset: function () {
                        this._hash = new WordArray.init(H.slice(0));
                    },

                    _doProcessBlock: function (M, offset) {
                        // Shortcut
                        var H = this._hash.words;

                        // Working variables
                        var a = H[0];
                        var b = H[1];
                        var c = H[2];
                        var d = H[3];
                        var e = H[4];
                        var f = H[5];
                        var g = H[6];
                        var h = H[7];

                        // Computation
                        for (var i = 0; i < 64; i++) {
                            if (i < 16) {
                                W[i] = M[offset + i] | 0;
                            } else {
                                var gamma0x = W[i - 15];
                                var gamma0 = ((gamma0x << 25) | (gamma0x >>> 7)) ^
                                    ((gamma0x << 14) | (gamma0x >>> 18)) ^
                                    (gamma0x >>> 3);

                                var gamma1x = W[i - 2];
                                var gamma1 = ((gamma1x << 15) | (gamma1x >>> 17)) ^
                                    ((gamma1x << 13) | (gamma1x >>> 19)) ^
                                    (gamma1x >>> 10);

                                W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16];
                            }

                            var ch = (e & f) ^ (~e & g);
                            var maj = (a & b) ^ (a & c) ^ (b & c);

                            var sigma0 = ((a << 30) | (a >>> 2)) ^ ((a << 19) | (a >>> 13)) ^ ((a << 10) | (a >>> 22));
                            var sigma1 = ((e << 26) | (e >>> 6)) ^ ((e << 21) | (e >>> 11)) ^ ((e << 7) | (e >>> 25));

                            var t1 = h + sigma1 + ch + K[i] + W[i];
                            var t2 = sigma0 + maj;

                            h = g;
                            g = f;
                            f = e;
                            e = (d + t1) | 0;
                            d = c;
                            c = b;
                            b = a;
                            a = (t1 + t2) | 0;
                        }

                        // Intermediate hash value
                        H[0] = (H[0] + a) | 0;
                        H[1] = (H[1] + b) | 0;
                        H[2] = (H[2] + c) | 0;
                        H[3] = (H[3] + d) | 0;
                        H[4] = (H[4] + e) | 0;
                        H[5] = (H[5] + f) | 0;
                        H[6] = (H[6] + g) | 0;
                        H[7] = (H[7] + h) | 0;
                    },

                    _doFinalize: function () {
                        // Shortcuts
                        var data = this._data;
                        var dataWords = data.words;

                        var nBitsTotal = this._nDataBytes * 8;
                        var nBitsLeft = data.sigBytes * 8;

                        // Add padding
                        dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
                        dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
                        dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
                        data.sigBytes = dataWords.length * 4;

                        // Hash final blocks
                        this._process();

                        // Return final computed hash
                        return this._hash;
                    },

                    clone: function () {
                        var clone = Hasher.clone.call(this);
                        clone._hash = this._hash.clone();

                        return clone;
                    }
                });

                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.SHA256('message');
                 *     var hash = CryptoJS.SHA256(wordArray);
                 */
                C.SHA256 = Hasher._createHelper(SHA256);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacSHA256(message, key);
                 */
                C.HmacSHA256 = Hasher._createHmacHelper(SHA256);
            }(Math));


            return CryptoJS.SHA256;

        }));

        /***/
    },

    /***/ 220:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(219));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./sha256"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var C_algo = C.algo;
                var SHA256 = C_algo.SHA256;

                /**
                 * SHA-224 hash algorithm.
                 */
                var SHA224 = C_algo.SHA224 = SHA256.extend({
                    _doReset: function () {
                        this._hash = new WordArray.init([
                            0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939,
                            0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4
                        ]);
                    },

                    _doFinalize: function () {
                        var hash = SHA256._doFinalize.call(this);

                        hash.sigBytes -= 4;

                        return hash;
                    }
                });

                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.SHA224('message');
                 *     var hash = CryptoJS.SHA224(wordArray);
                 */
                C.SHA224 = SHA256._createHelper(SHA224);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacSHA224(message, key);
                 */
                C.HmacSHA224 = SHA256._createHmacHelper(SHA224);
            }());


            return CryptoJS.SHA224;

        }));

        /***/
    },

    /***/ 221:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(213));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./x64-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var Hasher = C_lib.Hasher;
                var C_x64 = C.x64;
                var X64Word = C_x64.Word;
                var X64WordArray = C_x64.WordArray;
                var C_algo = C.algo;

                function X64Word_create() {
                    return X64Word.create.apply(X64Word, arguments);
                }

                // Constants
                var K = [
                    X64Word_create(0x428a2f98, 0xd728ae22), X64Word_create(0x71374491, 0x23ef65cd),
                    X64Word_create(0xb5c0fbcf, 0xec4d3b2f), X64Word_create(0xe9b5dba5, 0x8189dbbc),
                    X64Word_create(0x3956c25b, 0xf348b538), X64Word_create(0x59f111f1, 0xb605d019),
                    X64Word_create(0x923f82a4, 0xaf194f9b), X64Word_create(0xab1c5ed5, 0xda6d8118),
                    X64Word_create(0xd807aa98, 0xa3030242), X64Word_create(0x12835b01, 0x45706fbe),
                    X64Word_create(0x243185be, 0x4ee4b28c), X64Word_create(0x550c7dc3, 0xd5ffb4e2),
                    X64Word_create(0x72be5d74, 0xf27b896f), X64Word_create(0x80deb1fe, 0x3b1696b1),
                    X64Word_create(0x9bdc06a7, 0x25c71235), X64Word_create(0xc19bf174, 0xcf692694),
                    X64Word_create(0xe49b69c1, 0x9ef14ad2), X64Word_create(0xefbe4786, 0x384f25e3),
                    X64Word_create(0x0fc19dc6, 0x8b8cd5b5), X64Word_create(0x240ca1cc, 0x77ac9c65),
                    X64Word_create(0x2de92c6f, 0x592b0275), X64Word_create(0x4a7484aa, 0x6ea6e483),
                    X64Word_create(0x5cb0a9dc, 0xbd41fbd4), X64Word_create(0x76f988da, 0x831153b5),
                    X64Word_create(0x983e5152, 0xee66dfab), X64Word_create(0xa831c66d, 0x2db43210),
                    X64Word_create(0xb00327c8, 0x98fb213f), X64Word_create(0xbf597fc7, 0xbeef0ee4),
                    X64Word_create(0xc6e00bf3, 0x3da88fc2), X64Word_create(0xd5a79147, 0x930aa725),
                    X64Word_create(0x06ca6351, 0xe003826f), X64Word_create(0x14292967, 0x0a0e6e70),
                    X64Word_create(0x27b70a85, 0x46d22ffc), X64Word_create(0x2e1b2138, 0x5c26c926),
                    X64Word_create(0x4d2c6dfc, 0x5ac42aed), X64Word_create(0x53380d13, 0x9d95b3df),
                    X64Word_create(0x650a7354, 0x8baf63de), X64Word_create(0x766a0abb, 0x3c77b2a8),
                    X64Word_create(0x81c2c92e, 0x47edaee6), X64Word_create(0x92722c85, 0x1482353b),
                    X64Word_create(0xa2bfe8a1, 0x4cf10364), X64Word_create(0xa81a664b, 0xbc423001),
                    X64Word_create(0xc24b8b70, 0xd0f89791), X64Word_create(0xc76c51a3, 0x0654be30),
                    X64Word_create(0xd192e819, 0xd6ef5218), X64Word_create(0xd6990624, 0x5565a910),
                    X64Word_create(0xf40e3585, 0x5771202a), X64Word_create(0x106aa070, 0x32bbd1b8),
                    X64Word_create(0x19a4c116, 0xb8d2d0c8), X64Word_create(0x1e376c08, 0x5141ab53),
                    X64Word_create(0x2748774c, 0xdf8eeb99), X64Word_create(0x34b0bcb5, 0xe19b48a8),
                    X64Word_create(0x391c0cb3, 0xc5c95a63), X64Word_create(0x4ed8aa4a, 0xe3418acb),
                    X64Word_create(0x5b9cca4f, 0x7763e373), X64Word_create(0x682e6ff3, 0xd6b2b8a3),
                    X64Word_create(0x748f82ee, 0x5defb2fc), X64Word_create(0x78a5636f, 0x43172f60),
                    X64Word_create(0x84c87814, 0xa1f0ab72), X64Word_create(0x8cc70208, 0x1a6439ec),
                    X64Word_create(0x90befffa, 0x23631e28), X64Word_create(0xa4506ceb, 0xde82bde9),
                    X64Word_create(0xbef9a3f7, 0xb2c67915), X64Word_create(0xc67178f2, 0xe372532b),
                    X64Word_create(0xca273ece, 0xea26619c), X64Word_create(0xd186b8c7, 0x21c0c207),
                    X64Word_create(0xeada7dd6, 0xcde0eb1e), X64Word_create(0xf57d4f7f, 0xee6ed178),
                    X64Word_create(0x06f067aa, 0x72176fba), X64Word_create(0x0a637dc5, 0xa2c898a6),
                    X64Word_create(0x113f9804, 0xbef90dae), X64Word_create(0x1b710b35, 0x131c471b),
                    X64Word_create(0x28db77f5, 0x23047d84), X64Word_create(0x32caab7b, 0x40c72493),
                    X64Word_create(0x3c9ebe0a, 0x15c9bebc), X64Word_create(0x431d67c4, 0x9c100d4c),
                    X64Word_create(0x4cc5d4be, 0xcb3e42b6), X64Word_create(0x597f299c, 0xfc657e2a),
                    X64Word_create(0x5fcb6fab, 0x3ad6faec), X64Word_create(0x6c44198c, 0x4a475817)
                ];

                // Reusable objects
                var W = [];
                (function () {
                    for (var i = 0; i < 80; i++) {
                        W[i] = X64Word_create();
                    }
                }());

                /**
                 * SHA-512 hash algorithm.
                 */
                var SHA512 = C_algo.SHA512 = Hasher.extend({
                    _doReset: function () {
                        this._hash = new X64WordArray.init([
                            new X64Word.init(0x6a09e667, 0xf3bcc908), new X64Word.init(0xbb67ae85, 0x84caa73b),
                            new X64Word.init(0x3c6ef372, 0xfe94f82b), new X64Word.init(0xa54ff53a, 0x5f1d36f1),
                            new X64Word.init(0x510e527f, 0xade682d1), new X64Word.init(0x9b05688c, 0x2b3e6c1f),
                            new X64Word.init(0x1f83d9ab, 0xfb41bd6b), new X64Word.init(0x5be0cd19, 0x137e2179)
                        ]);
                    },

                    _doProcessBlock: function (M, offset) {
                        // Shortcuts
                        var H = this._hash.words;

                        var H0 = H[0];
                        var H1 = H[1];
                        var H2 = H[2];
                        var H3 = H[3];
                        var H4 = H[4];
                        var H5 = H[5];
                        var H6 = H[6];
                        var H7 = H[7];

                        var H0h = H0.high;
                        var H0l = H0.low;
                        var H1h = H1.high;
                        var H1l = H1.low;
                        var H2h = H2.high;
                        var H2l = H2.low;
                        var H3h = H3.high;
                        var H3l = H3.low;
                        var H4h = H4.high;
                        var H4l = H4.low;
                        var H5h = H5.high;
                        var H5l = H5.low;
                        var H6h = H6.high;
                        var H6l = H6.low;
                        var H7h = H7.high;
                        var H7l = H7.low;

                        // Working variables
                        var ah = H0h;
                        var al = H0l;
                        var bh = H1h;
                        var bl = H1l;
                        var ch = H2h;
                        var cl = H2l;
                        var dh = H3h;
                        var dl = H3l;
                        var eh = H4h;
                        var el = H4l;
                        var fh = H5h;
                        var fl = H5l;
                        var gh = H6h;
                        var gl = H6l;
                        var hh = H7h;
                        var hl = H7l;

                        // Rounds
                        for (var i = 0; i < 80; i++) {
                            // Shortcut
                            var Wi = W[i];

                            // Extend message
                            if (i < 16) {
                                var Wih = Wi.high = M[offset + i * 2] | 0;
                                var Wil = Wi.low = M[offset + i * 2 + 1] | 0;
                            } else {
                                // Gamma0
                                var gamma0x = W[i - 15];
                                var gamma0xh = gamma0x.high;
                                var gamma0xl = gamma0x.low;
                                var gamma0h = ((gamma0xh >>> 1) | (gamma0xl << 31)) ^ ((gamma0xh >>> 8) | (gamma0xl << 24)) ^ (gamma0xh >>> 7);
                                var gamma0l = ((gamma0xl >>> 1) | (gamma0xh << 31)) ^ ((gamma0xl >>> 8) | (gamma0xh << 24)) ^ ((gamma0xl >>> 7) | (gamma0xh << 25));

                                // Gamma1
                                var gamma1x = W[i - 2];
                                var gamma1xh = gamma1x.high;
                                var gamma1xl = gamma1x.low;
                                var gamma1h = ((gamma1xh >>> 19) | (gamma1xl << 13)) ^ ((gamma1xh << 3) | (gamma1xl >>> 29)) ^ (gamma1xh >>> 6);
                                var gamma1l = ((gamma1xl >>> 19) | (gamma1xh << 13)) ^ ((gamma1xl << 3) | (gamma1xh >>> 29)) ^ ((gamma1xl >>> 6) | (gamma1xh << 26));

                                // W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16]
                                var Wi7 = W[i - 7];
                                var Wi7h = Wi7.high;
                                var Wi7l = Wi7.low;

                                var Wi16 = W[i - 16];
                                var Wi16h = Wi16.high;
                                var Wi16l = Wi16.low;

                                var Wil = gamma0l + Wi7l;
                                var Wih = gamma0h + Wi7h + ((Wil >>> 0) < (gamma0l >>> 0) ? 1 : 0);
                                var Wil = Wil + gamma1l;
                                var Wih = Wih + gamma1h + ((Wil >>> 0) < (gamma1l >>> 0) ? 1 : 0);
                                var Wil = Wil + Wi16l;
                                var Wih = Wih + Wi16h + ((Wil >>> 0) < (Wi16l >>> 0) ? 1 : 0);

                                Wi.high = Wih;
                                Wi.low = Wil;
                            }

                            var chh = (eh & fh) ^ (~eh & gh);
                            var chl = (el & fl) ^ (~el & gl);
                            var majh = (ah & bh) ^ (ah & ch) ^ (bh & ch);
                            var majl = (al & bl) ^ (al & cl) ^ (bl & cl);

                            var sigma0h = ((ah >>> 28) | (al << 4)) ^ ((ah << 30) | (al >>> 2)) ^ ((ah << 25) | (al >>> 7));
                            var sigma0l = ((al >>> 28) | (ah << 4)) ^ ((al << 30) | (ah >>> 2)) ^ ((al << 25) | (ah >>> 7));
                            var sigma1h = ((eh >>> 14) | (el << 18)) ^ ((eh >>> 18) | (el << 14)) ^ ((eh << 23) | (el >>> 9));
                            var sigma1l = ((el >>> 14) | (eh << 18)) ^ ((el >>> 18) | (eh << 14)) ^ ((el << 23) | (eh >>> 9));

                            // t1 = h + sigma1 + ch + K[i] + W[i]
                            var Ki = K[i];
                            var Kih = Ki.high;
                            var Kil = Ki.low;

                            var t1l = hl + sigma1l;
                            var t1h = hh + sigma1h + ((t1l >>> 0) < (hl >>> 0) ? 1 : 0);
                            var t1l = t1l + chl;
                            var t1h = t1h + chh + ((t1l >>> 0) < (chl >>> 0) ? 1 : 0);
                            var t1l = t1l + Kil;
                            var t1h = t1h + Kih + ((t1l >>> 0) < (Kil >>> 0) ? 1 : 0);
                            var t1l = t1l + Wil;
                            var t1h = t1h + Wih + ((t1l >>> 0) < (Wil >>> 0) ? 1 : 0);

                            // t2 = sigma0 + maj
                            var t2l = sigma0l + majl;
                            var t2h = sigma0h + majh + ((t2l >>> 0) < (sigma0l >>> 0) ? 1 : 0);

                            // Update working variables
                            hh = gh;
                            hl = gl;
                            gh = fh;
                            gl = fl;
                            fh = eh;
                            fl = el;
                            el = (dl + t1l) | 0;
                            eh = (dh + t1h + ((el >>> 0) < (dl >>> 0) ? 1 : 0)) | 0;
                            dh = ch;
                            dl = cl;
                            ch = bh;
                            cl = bl;
                            bh = ah;
                            bl = al;
                            al = (t1l + t2l) | 0;
                            ah = (t1h + t2h + ((al >>> 0) < (t1l >>> 0) ? 1 : 0)) | 0;
                        }

                        // Intermediate hash value
                        H0l = H0.low = (H0l + al);
                        H0.high = (H0h + ah + ((H0l >>> 0) < (al >>> 0) ? 1 : 0));
                        H1l = H1.low = (H1l + bl);
                        H1.high = (H1h + bh + ((H1l >>> 0) < (bl >>> 0) ? 1 : 0));
                        H2l = H2.low = (H2l + cl);
                        H2.high = (H2h + ch + ((H2l >>> 0) < (cl >>> 0) ? 1 : 0));
                        H3l = H3.low = (H3l + dl);
                        H3.high = (H3h + dh + ((H3l >>> 0) < (dl >>> 0) ? 1 : 0));
                        H4l = H4.low = (H4l + el);
                        H4.high = (H4h + eh + ((H4l >>> 0) < (el >>> 0) ? 1 : 0));
                        H5l = H5.low = (H5l + fl);
                        H5.high = (H5h + fh + ((H5l >>> 0) < (fl >>> 0) ? 1 : 0));
                        H6l = H6.low = (H6l + gl);
                        H6.high = (H6h + gh + ((H6l >>> 0) < (gl >>> 0) ? 1 : 0));
                        H7l = H7.low = (H7l + hl);
                        H7.high = (H7h + hh + ((H7l >>> 0) < (hl >>> 0) ? 1 : 0));
                    },

                    _doFinalize: function () {
                        // Shortcuts
                        var data = this._data;
                        var dataWords = data.words;

                        var nBitsTotal = this._nDataBytes * 8;
                        var nBitsLeft = data.sigBytes * 8;

                        // Add padding
                        dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
                        dataWords[(((nBitsLeft + 128) >>> 10) << 5) + 30] = Math.floor(nBitsTotal / 0x100000000);
                        dataWords[(((nBitsLeft + 128) >>> 10) << 5) + 31] = nBitsTotal;
                        data.sigBytes = dataWords.length * 4;

                        // Hash final blocks
                        this._process();

                        // Convert hash to 32-bit word array before returning
                        var hash = this._hash.toX32();

                        // Return final computed hash
                        return hash;
                    },

                    clone: function () {
                        var clone = Hasher.clone.call(this);
                        clone._hash = this._hash.clone();

                        return clone;
                    },

                    blockSize: 1024 / 32
                });

                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.SHA512('message');
                 *     var hash = CryptoJS.SHA512(wordArray);
                 */
                C.SHA512 = Hasher._createHelper(SHA512);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacSHA512(message, key);
                 */
                C.HmacSHA512 = Hasher._createHmacHelper(SHA512);
            }());


            return CryptoJS.SHA512;

        }));

        /***/
    },

    /***/ 222:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(213), __webpack_require__(221));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./x64-core", "./sha512"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_x64 = C.x64;
                var X64Word = C_x64.Word;
                var X64WordArray = C_x64.WordArray;
                var C_algo = C.algo;
                var SHA512 = C_algo.SHA512;

                /**
                 * SHA-384 hash algorithm.
                 */
                var SHA384 = C_algo.SHA384 = SHA512.extend({
                    _doReset: function () {
                        this._hash = new X64WordArray.init([
                            new X64Word.init(0xcbbb9d5d, 0xc1059ed8), new X64Word.init(0x629a292a, 0x367cd507),
                            new X64Word.init(0x9159015a, 0x3070dd17), new X64Word.init(0x152fecd8, 0xf70e5939),
                            new X64Word.init(0x67332667, 0xffc00b31), new X64Word.init(0x8eb44a87, 0x68581511),
                            new X64Word.init(0xdb0c2e0d, 0x64f98fa7), new X64Word.init(0x47b5481d, 0xbefa4fa4)
                        ]);
                    },

                    _doFinalize: function () {
                        var hash = SHA512._doFinalize.call(this);

                        hash.sigBytes -= 16;

                        return hash;
                    }
                });

                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.SHA384('message');
                 *     var hash = CryptoJS.SHA384(wordArray);
                 */
                C.SHA384 = SHA512._createHelper(SHA384);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacSHA384(message, key);
                 */
                C.HmacSHA384 = SHA512._createHmacHelper(SHA384);
            }());


            return CryptoJS.SHA384;

        }));

        /***/
    },

    /***/ 223:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(213));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./x64-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function (Math) {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var Hasher = C_lib.Hasher;
                var C_x64 = C.x64;
                var X64Word = C_x64.Word;
                var C_algo = C.algo;

                // Constants tables
                var RHO_OFFSETS = [];
                var PI_INDEXES = [];
                var ROUND_CONSTANTS = [];

                // Compute Constants
                (function () {
                    // Compute rho offset constants
                    var x = 1, y = 0;
                    for (var t = 0; t < 24; t++) {
                        RHO_OFFSETS[x + 5 * y] = ((t + 1) * (t + 2) / 2) % 64;

                        var newX = y % 5;
                        var newY = (2 * x + 3 * y) % 5;
                        x = newX;
                        y = newY;
                    }

                    // Compute pi index constants
                    for (var x = 0; x < 5; x++) {
                        for (var y = 0; y < 5; y++) {
                            PI_INDEXES[x + 5 * y] = y + ((2 * x + 3 * y) % 5) * 5;
                        }
                    }

                    // Compute round constants
                    var LFSR = 0x01;
                    for (var i = 0; i < 24; i++) {
                        var roundConstantMsw = 0;
                        var roundConstantLsw = 0;

                        for (var j = 0; j < 7; j++) {
                            if (LFSR & 0x01) {
                                var bitPosition = (1 << j) - 1;
                                if (bitPosition < 32) {
                                    roundConstantLsw ^= 1 << bitPosition;
                                } else /* if (bitPosition >= 32) */ {
                                    roundConstantMsw ^= 1 << (bitPosition - 32);
                                }
                            }

                            // Compute next LFSR
                            if (LFSR & 0x80) {
                                // Primitive polynomial over GF(2): x^8 + x^6 + x^5 + x^4 + 1
                                LFSR = (LFSR << 1) ^ 0x71;
                            } else {
                                LFSR <<= 1;
                            }
                        }

                        ROUND_CONSTANTS[i] = X64Word.create(roundConstantMsw, roundConstantLsw);
                    }
                }());

                // Reusable objects for temporary values
                var T = [];
                (function () {
                    for (var i = 0; i < 25; i++) {
                        T[i] = X64Word.create();
                    }
                }());

                /**
                 * SHA-3 hash algorithm.
                 */
                var SHA3 = C_algo.SHA3 = Hasher.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {number} outputLength
                     *   The desired number of bits in the output hash.
                     *   Only values permitted are: 224, 256, 384, 512.
                     *   Default: 512
                     */
                    cfg: Hasher.cfg.extend({
                        outputLength: 512
                    }),

                    _doReset: function () {
                        var state = this._state = []
                        for (var i = 0; i < 25; i++) {
                            state[i] = new X64Word.init();
                        }

                        this.blockSize = (1600 - 2 * this.cfg.outputLength) / 32;
                    },

                    _doProcessBlock: function (M, offset) {
                        // Shortcuts
                        var state = this._state;
                        var nBlockSizeLanes = this.blockSize / 2;

                        // Absorb
                        for (var i = 0; i < nBlockSizeLanes; i++) {
                            // Shortcuts
                            var M2i = M[offset + 2 * i];
                            var M2i1 = M[offset + 2 * i + 1];

                            // Swap endian
                            M2i = (
                                (((M2i << 8) | (M2i >>> 24)) & 0x00ff00ff) |
                                (((M2i << 24) | (M2i >>> 8)) & 0xff00ff00)
                            );
                            M2i1 = (
                                (((M2i1 << 8) | (M2i1 >>> 24)) & 0x00ff00ff) |
                                (((M2i1 << 24) | (M2i1 >>> 8)) & 0xff00ff00)
                            );

                            // Absorb message into state
                            var lane = state[i];
                            lane.high ^= M2i1;
                            lane.low ^= M2i;
                        }

                        // Rounds
                        for (var round = 0; round < 24; round++) {
                            // Theta
                            for (var x = 0; x < 5; x++) {
                                // Mix column lanes
                                var tMsw = 0, tLsw = 0;
                                for (var y = 0; y < 5; y++) {
                                    var lane = state[x + 5 * y];
                                    tMsw ^= lane.high;
                                    tLsw ^= lane.low;
                                }

                                // Temporary values
                                var Tx = T[x];
                                Tx.high = tMsw;
                                Tx.low = tLsw;
                            }
                            for (var x = 0; x < 5; x++) {
                                // Shortcuts
                                var Tx4 = T[(x + 4) % 5];
                                var Tx1 = T[(x + 1) % 5];
                                var Tx1Msw = Tx1.high;
                                var Tx1Lsw = Tx1.low;

                                // Mix surrounding columns
                                var tMsw = Tx4.high ^ ((Tx1Msw << 1) | (Tx1Lsw >>> 31));
                                var tLsw = Tx4.low ^ ((Tx1Lsw << 1) | (Tx1Msw >>> 31));
                                for (var y = 0; y < 5; y++) {
                                    var lane = state[x + 5 * y];
                                    lane.high ^= tMsw;
                                    lane.low ^= tLsw;
                                }
                            }

                            // Rho Pi
                            for (var laneIndex = 1; laneIndex < 25; laneIndex++) {
                                // Shortcuts
                                var lane = state[laneIndex];
                                var laneMsw = lane.high;
                                var laneLsw = lane.low;
                                var rhoOffset = RHO_OFFSETS[laneIndex];

                                // Rotate lanes
                                if (rhoOffset < 32) {
                                    var tMsw = (laneMsw << rhoOffset) | (laneLsw >>> (32 - rhoOffset));
                                    var tLsw = (laneLsw << rhoOffset) | (laneMsw >>> (32 - rhoOffset));
                                } else /* if (rhoOffset >= 32) */ {
                                    var tMsw = (laneLsw << (rhoOffset - 32)) | (laneMsw >>> (64 - rhoOffset));
                                    var tLsw = (laneMsw << (rhoOffset - 32)) | (laneLsw >>> (64 - rhoOffset));
                                }

                                // Transpose lanes
                                var TPiLane = T[PI_INDEXES[laneIndex]];
                                TPiLane.high = tMsw;
                                TPiLane.low = tLsw;
                            }

                            // Rho pi at x = y = 0
                            var T0 = T[0];
                            var state0 = state[0];
                            T0.high = state0.high;
                            T0.low = state0.low;

                            // Chi
                            for (var x = 0; x < 5; x++) {
                                for (var y = 0; y < 5; y++) {
                                    // Shortcuts
                                    var laneIndex = x + 5 * y;
                                    var lane = state[laneIndex];
                                    var TLane = T[laneIndex];
                                    var Tx1Lane = T[((x + 1) % 5) + 5 * y];
                                    var Tx2Lane = T[((x + 2) % 5) + 5 * y];

                                    // Mix rows
                                    lane.high = TLane.high ^ (~Tx1Lane.high & Tx2Lane.high);
                                    lane.low = TLane.low ^ (~Tx1Lane.low & Tx2Lane.low);
                                }
                            }

                            // Iota
                            var lane = state[0];
                            var roundConstant = ROUND_CONSTANTS[round];
                            lane.high ^= roundConstant.high;
                            lane.low ^= roundConstant.low;
                            ;
                        }
                    },

                    _doFinalize: function () {
                        // Shortcuts
                        var data = this._data;
                        var dataWords = data.words;
                        var nBitsTotal = this._nDataBytes * 8;
                        var nBitsLeft = data.sigBytes * 8;
                        var blockSizeBits = this.blockSize * 32;

                        // Add padding
                        dataWords[nBitsLeft >>> 5] |= 0x1 << (24 - nBitsLeft % 32);
                        dataWords[((Math.ceil((nBitsLeft + 1) / blockSizeBits) * blockSizeBits) >>> 5) - 1] |= 0x80;
                        data.sigBytes = dataWords.length * 4;

                        // Hash final blocks
                        this._process();

                        // Shortcuts
                        var state = this._state;
                        var outputLengthBytes = this.cfg.outputLength / 8;
                        var outputLengthLanes = outputLengthBytes / 8;

                        // Squeeze
                        var hashWords = [];
                        for (var i = 0; i < outputLengthLanes; i++) {
                            // Shortcuts
                            var lane = state[i];
                            var laneMsw = lane.high;
                            var laneLsw = lane.low;

                            // Swap endian
                            laneMsw = (
                                (((laneMsw << 8) | (laneMsw >>> 24)) & 0x00ff00ff) |
                                (((laneMsw << 24) | (laneMsw >>> 8)) & 0xff00ff00)
                            );
                            laneLsw = (
                                (((laneLsw << 8) | (laneLsw >>> 24)) & 0x00ff00ff) |
                                (((laneLsw << 24) | (laneLsw >>> 8)) & 0xff00ff00)
                            );

                            // Squeeze state to retrieve hash
                            hashWords.push(laneLsw);
                            hashWords.push(laneMsw);
                        }

                        // Return final computed hash
                        return new WordArray.init(hashWords, outputLengthBytes);
                    },

                    clone: function () {
                        var clone = Hasher.clone.call(this);

                        var state = clone._state = this._state.slice(0);
                        for (var i = 0; i < 25; i++) {
                            state[i] = state[i].clone();
                        }

                        return clone;
                    }
                });

                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.SHA3('message');
                 *     var hash = CryptoJS.SHA3(wordArray);
                 */
                C.SHA3 = Hasher._createHelper(SHA3);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacSHA3(message, key);
                 */
                C.HmacSHA3 = Hasher._createHmacHelper(SHA3);
            }(Math));


            return CryptoJS.SHA3;

        }));

        /***/
    },

    /***/ 224:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /** @preserve
             (c) 2012 by Cédric Mesnil. All rights reserved.

             Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:

             - Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
             - Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.

             THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
             */

            (function (Math) {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var Hasher = C_lib.Hasher;
                var C_algo = C.algo;

                // Constants table
                var _zl = WordArray.create([
                    0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15,
                    7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8,
                    3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12,
                    1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2,
                    4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13]);
                var _zr = WordArray.create([
                    5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12,
                    6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2,
                    15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13,
                    8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14,
                    12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11]);
                var _sl = WordArray.create([
                    11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8,
                    7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12,
                    11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5,
                    11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12,
                    9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6]);
                var _sr = WordArray.create([
                    8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6,
                    9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11,
                    9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5,
                    15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8,
                    8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11]);

                var _hl = WordArray.create([0x00000000, 0x5A827999, 0x6ED9EBA1, 0x8F1BBCDC, 0xA953FD4E]);
                var _hr = WordArray.create([0x50A28BE6, 0x5C4DD124, 0x6D703EF3, 0x7A6D76E9, 0x00000000]);

                /**
                 * RIPEMD160 hash algorithm.
                 */
                var RIPEMD160 = C_algo.RIPEMD160 = Hasher.extend({
                    _doReset: function () {
                        this._hash = WordArray.create([0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0]);
                    },

                    _doProcessBlock: function (M, offset) {

                        // Swap endian
                        for (var i = 0; i < 16; i++) {
                            // Shortcuts
                            var offset_i = offset + i;
                            var M_offset_i = M[offset_i];

                            // Swap
                            M[offset_i] = (
                                (((M_offset_i << 8) | (M_offset_i >>> 24)) & 0x00ff00ff) |
                                (((M_offset_i << 24) | (M_offset_i >>> 8)) & 0xff00ff00)
                            );
                        }
                        // Shortcut
                        var H = this._hash.words;
                        var hl = _hl.words;
                        var hr = _hr.words;
                        var zl = _zl.words;
                        var zr = _zr.words;
                        var sl = _sl.words;
                        var sr = _sr.words;

                        // Working variables
                        var al, bl, cl, dl, el;
                        var ar, br, cr, dr, er;

                        ar = al = H[0];
                        br = bl = H[1];
                        cr = cl = H[2];
                        dr = dl = H[3];
                        er = el = H[4];
                        // Computation
                        var t;
                        for (var i = 0; i < 80; i += 1) {
                            t = (al + M[offset + zl[i]]) | 0;
                            if (i < 16) {
                                t += f1(bl, cl, dl) + hl[0];
                            } else if (i < 32) {
                                t += f2(bl, cl, dl) + hl[1];
                            } else if (i < 48) {
                                t += f3(bl, cl, dl) + hl[2];
                            } else if (i < 64) {
                                t += f4(bl, cl, dl) + hl[3];
                            } else {// if (i<80) {
                                t += f5(bl, cl, dl) + hl[4];
                            }
                            t = t | 0;
                            t = rotl(t, sl[i]);
                            t = (t + el) | 0;
                            al = el;
                            el = dl;
                            dl = rotl(cl, 10);
                            cl = bl;
                            bl = t;

                            t = (ar + M[offset + zr[i]]) | 0;
                            if (i < 16) {
                                t += f5(br, cr, dr) + hr[0];
                            } else if (i < 32) {
                                t += f4(br, cr, dr) + hr[1];
                            } else if (i < 48) {
                                t += f3(br, cr, dr) + hr[2];
                            } else if (i < 64) {
                                t += f2(br, cr, dr) + hr[3];
                            } else {// if (i<80) {
                                t += f1(br, cr, dr) + hr[4];
                            }
                            t = t | 0;
                            t = rotl(t, sr[i]);
                            t = (t + er) | 0;
                            ar = er;
                            er = dr;
                            dr = rotl(cr, 10);
                            cr = br;
                            br = t;
                        }
                        // Intermediate hash value
                        t = (H[1] + cl + dr) | 0;
                        H[1] = (H[2] + dl + er) | 0;
                        H[2] = (H[3] + el + ar) | 0;
                        H[3] = (H[4] + al + br) | 0;
                        H[4] = (H[0] + bl + cr) | 0;
                        H[0] = t;
                    },

                    _doFinalize: function () {
                        // Shortcuts
                        var data = this._data;
                        var dataWords = data.words;

                        var nBitsTotal = this._nDataBytes * 8;
                        var nBitsLeft = data.sigBytes * 8;

                        // Add padding
                        dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
                        dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = (
                            (((nBitsTotal << 8) | (nBitsTotal >>> 24)) & 0x00ff00ff) |
                            (((nBitsTotal << 24) | (nBitsTotal >>> 8)) & 0xff00ff00)
                        );
                        data.sigBytes = (dataWords.length + 1) * 4;

                        // Hash final blocks
                        this._process();

                        // Shortcuts
                        var hash = this._hash;
                        var H = hash.words;

                        // Swap endian
                        for (var i = 0; i < 5; i++) {
                            // Shortcut
                            var H_i = H[i];

                            // Swap
                            H[i] = (((H_i << 8) | (H_i >>> 24)) & 0x00ff00ff) |
                                (((H_i << 24) | (H_i >>> 8)) & 0xff00ff00);
                        }

                        // Return final computed hash
                        return hash;
                    },

                    clone: function () {
                        var clone = Hasher.clone.call(this);
                        clone._hash = this._hash.clone();

                        return clone;
                    }
                });


                function f1(x, y, z) {
                    return ((x) ^ (y) ^ (z));

                }

                function f2(x, y, z) {
                    return (((x) & (y)) | ((~x) & (z)));
                }

                function f3(x, y, z) {
                    return (((x) | (~(y))) ^ (z));
                }

                function f4(x, y, z) {
                    return (((x) & (z)) | ((y) & (~(z))));
                }

                function f5(x, y, z) {
                    return ((x) ^ ((y) | (~(z))));

                }

                function rotl(x, n) {
                    return (x << n) | (x >>> (32 - n));
                }


                /**
                 * Shortcut function to the hasher's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 *
                 * @return {WordArray} The hash.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hash = CryptoJS.RIPEMD160('message');
                 *     var hash = CryptoJS.RIPEMD160(wordArray);
                 */
                C.RIPEMD160 = Hasher._createHelper(RIPEMD160);

                /**
                 * Shortcut function to the HMAC's object interface.
                 *
                 * @param {WordArray|string} message The message to hash.
                 * @param {WordArray|string} key The secret key.
                 *
                 * @return {WordArray} The HMAC.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var hmac = CryptoJS.HmacRIPEMD160(message, key);
                 */
                C.HmacRIPEMD160 = Hasher._createHmacHelper(RIPEMD160);
            }(Math));


            return CryptoJS.RIPEMD160;

        }));

        /***/
    },

    /***/ 225:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var Base = C_lib.Base;
                var C_enc = C.enc;
                var Utf8 = C_enc.Utf8;
                var C_algo = C.algo;

                /**
                 * HMAC algorithm.
                 */
                var HMAC = C_algo.HMAC = Base.extend({
                    /**
                     * Initializes a newly created HMAC.
                     *
                     * @param {Hasher} hasher The hash algorithm to use.
                     * @param {WordArray|string} key The secret key.
                     *
                     * @example
                     *
                     *     var hmacHasher = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, key);
                     */
                    init: function (hasher, key) {
                        // Init hasher
                        hasher = this._hasher = new hasher.init();

                        // Convert string to WordArray, else assume WordArray already
                        if (typeof key == 'string') {
                            key = Utf8.parse(key);
                        }

                        // Shortcuts
                        var hasherBlockSize = hasher.blockSize;
                        var hasherBlockSizeBytes = hasherBlockSize * 4;

                        // Allow arbitrary length keys
                        if (key.sigBytes > hasherBlockSizeBytes) {
                            key = hasher.finalize(key);
                        }

                        // Clamp excess bits
                        key.clamp();

                        // Clone key for inner and outer pads
                        var oKey = this._oKey = key.clone();
                        var iKey = this._iKey = key.clone();

                        // Shortcuts
                        var oKeyWords = oKey.words;
                        var iKeyWords = iKey.words;

                        // XOR keys with pad constants
                        for (var i = 0; i < hasherBlockSize; i++) {
                            oKeyWords[i] ^= 0x5c5c5c5c;
                            iKeyWords[i] ^= 0x36363636;
                        }
                        oKey.sigBytes = iKey.sigBytes = hasherBlockSizeBytes;

                        // Set initial values
                        this.reset();
                    },

                    /**
                     * Resets this HMAC to its initial state.
                     *
                     * @example
                     *
                     *     hmacHasher.reset();
                     */
                    reset: function () {
                        // Shortcut
                        var hasher = this._hasher;

                        // Reset
                        hasher.reset();
                        hasher.update(this._iKey);
                    },

                    /**
                     * Updates this HMAC with a message.
                     *
                     * @param {WordArray|string} messageUpdate The message to append.
                     *
                     * @return {HMAC} This HMAC instance.
                     *
                     * @example
                     *
                     *     hmacHasher.update('message');
                     *     hmacHasher.update(wordArray);
                     */
                    update: function (messageUpdate) {
                        this._hasher.update(messageUpdate);

                        // Chainable
                        return this;
                    },

                    /**
                     * Finalizes the HMAC computation.
                     * Note that the finalize operation is effectively a destructive, read-once operation.
                     *
                     * @param {WordArray|string} messageUpdate (Optional) A final message update.
                     *
                     * @return {WordArray} The HMAC.
                     *
                     * @example
                     *
                     *     var hmac = hmacHasher.finalize();
                     *     var hmac = hmacHasher.finalize('message');
                     *     var hmac = hmacHasher.finalize(wordArray);
                     */
                    finalize: function (messageUpdate) {
                        // Shortcut
                        var hasher = this._hasher;

                        // Compute HMAC
                        var innerHash = hasher.finalize(messageUpdate);
                        hasher.reset();
                        var hmac = hasher.finalize(this._oKey.clone().concat(innerHash));

                        return hmac;
                    }
                });
            }());


        }));

        /***/
    },

    /***/ 226:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(218), __webpack_require__(225));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./sha1", "./hmac"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var Base = C_lib.Base;
                var WordArray = C_lib.WordArray;
                var C_algo = C.algo;
                var SHA1 = C_algo.SHA1;
                var HMAC = C_algo.HMAC;

                /**
                 * Password-Based Key Derivation Function 2 algorithm.
                 */
                var PBKDF2 = C_algo.PBKDF2 = Base.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {number} keySize The key size in words to generate. Default: 4 (128 bits)
                     * @property {Hasher} hasher The hasher to use. Default: SHA1
                     * @property {number} iterations The number of iterations to perform. Default: 1
                     */
                    cfg: Base.extend({
                        keySize: 128 / 32,
                        hasher: SHA1,
                        iterations: 1
                    }),

                    /**
                     * Initializes a newly created key derivation function.
                     *
                     * @param {Object} cfg (Optional) The configuration options to use for the derivation.
                     *
                     * @example
                     *
                     *     var kdf = CryptoJS.algo.PBKDF2.create();
                     *     var kdf = CryptoJS.algo.PBKDF2.create({ keySize: 8 });
                     *     var kdf = CryptoJS.algo.PBKDF2.create({ keySize: 8, iterations: 1000 });
                     */
                    init: function (cfg) {
                        this.cfg = this.cfg.extend(cfg);
                    },

                    /**
                     * Computes the Password-Based Key Derivation Function 2.
                     *
                     * @param {WordArray|string} password The password.
                     * @param {WordArray|string} salt A salt.
                     *
                     * @return {WordArray} The derived key.
                     *
                     * @example
                     *
                     *     var key = kdf.compute(password, salt);
                     */
                    compute: function (password, salt) {
                        // Shortcut
                        var cfg = this.cfg;

                        // Init HMAC
                        var hmac = HMAC.create(cfg.hasher, password);

                        // Initial values
                        var derivedKey = WordArray.create();
                        var blockIndex = WordArray.create([0x00000001]);

                        // Shortcuts
                        var derivedKeyWords = derivedKey.words;
                        var blockIndexWords = blockIndex.words;
                        var keySize = cfg.keySize;
                        var iterations = cfg.iterations;

                        // Generate key
                        while (derivedKeyWords.length < keySize) {
                            var block = hmac.update(salt).finalize(blockIndex);
                            hmac.reset();

                            // Shortcuts
                            var blockWords = block.words;
                            var blockWordsLength = blockWords.length;

                            // Iterations
                            var intermediate = block;
                            for (var i = 1; i < iterations; i++) {
                                intermediate = hmac.finalize(intermediate);
                                hmac.reset();

                                // Shortcut
                                var intermediateWords = intermediate.words;

                                // XOR intermediate with block
                                for (var j = 0; j < blockWordsLength; j++) {
                                    blockWords[j] ^= intermediateWords[j];
                                }
                            }

                            derivedKey.concat(block);
                            blockIndexWords[0]++;
                        }
                        derivedKey.sigBytes = keySize * 4;

                        return derivedKey;
                    }
                });

                /**
                 * Computes the Password-Based Key Derivation Function 2.
                 *
                 * @param {WordArray|string} password The password.
                 * @param {WordArray|string} salt A salt.
                 * @param {Object} cfg (Optional) The configuration options to use for this computation.
                 *
                 * @return {WordArray} The derived key.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var key = CryptoJS.PBKDF2(password, salt);
                 *     var key = CryptoJS.PBKDF2(password, salt, { keySize: 8 });
                 *     var key = CryptoJS.PBKDF2(password, salt, { keySize: 8, iterations: 1000 });
                 */
                C.PBKDF2 = function (password, salt, cfg) {
                    return PBKDF2.create(cfg).compute(password, salt);
                };
            }());


            return CryptoJS.PBKDF2;

        }));

        /***/
    },

    /***/ 227:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(218), __webpack_require__(225));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./sha1", "./hmac"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var Base = C_lib.Base;
                var WordArray = C_lib.WordArray;
                var C_algo = C.algo;
                var MD5 = C_algo.MD5;

                /**
                 * This key derivation function is meant to conform with EVP_BytesToKey.
                 * www.openssl.org/docs/crypto/EVP_BytesToKey.html
                 */
                var EvpKDF = C_algo.EvpKDF = Base.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {number} keySize The key size in words to generate. Default: 4 (128 bits)
                     * @property {Hasher} hasher The hash algorithm to use. Default: MD5
                     * @property {number} iterations The number of iterations to perform. Default: 1
                     */
                    cfg: Base.extend({
                        keySize: 128 / 32,
                        hasher: MD5,
                        iterations: 1
                    }),

                    /**
                     * Initializes a newly created key derivation function.
                     *
                     * @param {Object} cfg (Optional) The configuration options to use for the derivation.
                     *
                     * @example
                     *
                     *     var kdf = CryptoJS.algo.EvpKDF.create();
                     *     var kdf = CryptoJS.algo.EvpKDF.create({ keySize: 8 });
                     *     var kdf = CryptoJS.algo.EvpKDF.create({ keySize: 8, iterations: 1000 });
                     */
                    init: function (cfg) {
                        this.cfg = this.cfg.extend(cfg);
                    },

                    /**
                     * Derives a key from a password.
                     *
                     * @param {WordArray|string} password The password.
                     * @param {WordArray|string} salt A salt.
                     *
                     * @return {WordArray} The derived key.
                     *
                     * @example
                     *
                     *     var key = kdf.compute(password, salt);
                     */
                    compute: function (password, salt) {
                        // Shortcut
                        var cfg = this.cfg;

                        // Init hasher
                        var hasher = cfg.hasher.create();

                        // Initial values
                        var derivedKey = WordArray.create();

                        // Shortcuts
                        var derivedKeyWords = derivedKey.words;
                        var keySize = cfg.keySize;
                        var iterations = cfg.iterations;

                        // Generate key
                        while (derivedKeyWords.length < keySize) {
                            if (block) {
                                hasher.update(block);
                            }
                            var block = hasher.update(password).finalize(salt);
                            hasher.reset();

                            // Iterations
                            for (var i = 1; i < iterations; i++) {
                                block = hasher.finalize(block);
                                hasher.reset();
                            }

                            derivedKey.concat(block);
                        }
                        derivedKey.sigBytes = keySize * 4;

                        return derivedKey;
                    }
                });

                /**
                 * Derives a key from a password.
                 *
                 * @param {WordArray|string} password The password.
                 * @param {WordArray|string} salt A salt.
                 * @param {Object} cfg (Optional) The configuration options to use for this computation.
                 *
                 * @return {WordArray} The derived key.
                 *
                 * @static
                 *
                 * @example
                 *
                 *     var key = CryptoJS.EvpKDF(password, salt);
                 *     var key = CryptoJS.EvpKDF(password, salt, { keySize: 8 });
                 *     var key = CryptoJS.EvpKDF(password, salt, { keySize: 8, iterations: 1000 });
                 */
                C.EvpKDF = function (password, salt, cfg) {
                    return EvpKDF.create(cfg).compute(password, salt);
                };
            }());


            return CryptoJS.EvpKDF;

        }));

        /***/
    },

    /***/ 228:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(227));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./evpkdf"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * Cipher core components.
             */
            CryptoJS.lib.Cipher || (function (undefined) {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var Base = C_lib.Base;
                var WordArray = C_lib.WordArray;
                var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm;
                var C_enc = C.enc;
                var Utf8 = C_enc.Utf8;
                var Base64 = C_enc.Base64;
                var C_algo = C.algo;
                var EvpKDF = C_algo.EvpKDF;

                /**
                 * Abstract base cipher template.
                 *
                 * @property {number} keySize This cipher's key size. Default: 4 (128 bits)
                 * @property {number} ivSize This cipher's IV size. Default: 4 (128 bits)
                 * @property {number} _ENC_XFORM_MODE A constant representing encryption mode.
                 * @property {number} _DEC_XFORM_MODE A constant representing decryption mode.
                 */
                var Cipher = C_lib.Cipher = BufferedBlockAlgorithm.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {WordArray} iv The IV to use for this operation.
                     */
                    cfg: Base.extend(),

                    /**
                     * Creates this cipher in encryption mode.
                     *
                     * @param {WordArray} key The key.
                     * @param {Object} cfg (Optional) The configuration options to use for this operation.
                     *
                     * @return {Cipher} A cipher instance.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var cipher = CryptoJS.algo.AES.createEncryptor(keyWordArray, { iv: ivWordArray });
                     */
                    createEncryptor: function (key, cfg) {
                        return this.create(this._ENC_XFORM_MODE, key, cfg);
                    },

                    /**
                     * Creates this cipher in decryption mode.
                     *
                     * @param {WordArray} key The key.
                     * @param {Object} cfg (Optional) The configuration options to use for this operation.
                     *
                     * @return {Cipher} A cipher instance.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var cipher = CryptoJS.algo.AES.createDecryptor(keyWordArray, { iv: ivWordArray });
                     */
                    createDecryptor: function (key, cfg) {
                        return this.create(this._DEC_XFORM_MODE, key, cfg);
                    },

                    /**
                     * Initializes a newly created cipher.
                     *
                     * @param {number} xformMode Either the encryption or decryption transormation mode constant.
                     * @param {WordArray} key The key.
                     * @param {Object} cfg (Optional) The configuration options to use for this operation.
                     *
                     * @example
                     *
                     *     var cipher = CryptoJS.algo.AES.create(CryptoJS.algo.AES._ENC_XFORM_MODE, keyWordArray, { iv: ivWordArray });
                     */
                    init: function (xformMode, key, cfg) {
                        // Apply config defaults
                        this.cfg = this.cfg.extend(cfg);

                        // Store transform mode and key
                        this._xformMode = xformMode;
                        this._key = key;

                        // Set initial values
                        this.reset();
                    },

                    /**
                     * Resets this cipher to its initial state.
                     *
                     * @example
                     *
                     *     cipher.reset();
                     */
                    reset: function () {
                        // Reset data buffer
                        BufferedBlockAlgorithm.reset.call(this);

                        // Perform concrete-cipher logic
                        this._doReset();
                    },

                    /**
                     * Adds data to be encrypted or decrypted.
                     *
                     * @param {WordArray|string} dataUpdate The data to encrypt or decrypt.
                     *
                     * @return {WordArray} The data after processing.
                     *
                     * @example
                     *
                     *     var encrypted = cipher.process('data');
                     *     var encrypted = cipher.process(wordArray);
                     */
                    process: function (dataUpdate) {
                        // Append
                        this._append(dataUpdate);

                        // Process available blocks
                        return this._process();
                    },

                    /**
                     * Finalizes the encryption or decryption process.
                     * Note that the finalize operation is effectively a destructive, read-once operation.
                     *
                     * @param {WordArray|string} dataUpdate The final data to encrypt or decrypt.
                     *
                     * @return {WordArray} The data after final processing.
                     *
                     * @example
                     *
                     *     var encrypted = cipher.finalize();
                     *     var encrypted = cipher.finalize('data');
                     *     var encrypted = cipher.finalize(wordArray);
                     */
                    finalize: function (dataUpdate) {
                        // Final data update
                        if (dataUpdate) {
                            this._append(dataUpdate);
                        }

                        // Perform concrete-cipher logic
                        var finalProcessedData = this._doFinalize();

                        return finalProcessedData;
                    },

                    keySize: 128 / 32,

                    ivSize: 128 / 32,

                    _ENC_XFORM_MODE: 1,

                    _DEC_XFORM_MODE: 2,

                    /**
                     * Creates shortcut functions to a cipher's object interface.
                     *
                     * @param {Cipher} cipher The cipher to create a helper for.
                     *
                     * @return {Object} An object with encrypt and decrypt shortcut functions.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var AES = CryptoJS.lib.Cipher._createHelper(CryptoJS.algo.AES);
                     */
                    _createHelper: (function () {
                        function selectCipherStrategy(key) {
                            if (typeof key == 'string') {
                                return PasswordBasedCipher;
                            } else {
                                return SerializableCipher;
                            }
                        }

                        return function (cipher) {
                            return {
                                encrypt: function (message, key, cfg) {
                                    return selectCipherStrategy(key).encrypt(cipher, message, key, cfg);
                                },

                                decrypt: function (ciphertext, key, cfg) {
                                    return selectCipherStrategy(key).decrypt(cipher, ciphertext, key, cfg);
                                }
                            };
                        };
                    }())
                });

                /**
                 * Abstract base stream cipher template.
                 *
                 * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 1 (32 bits)
                 */
                var StreamCipher = C_lib.StreamCipher = Cipher.extend({
                    _doFinalize: function () {
                        // Process partial blocks
                        var finalProcessedBlocks = this._process(!!'flush');

                        return finalProcessedBlocks;
                    },

                    blockSize: 1
                });

                /**
                 * Mode namespace.
                 */
                var C_mode = C.mode = {};

                /**
                 * Abstract base block cipher mode template.
                 */
                var BlockCipherMode = C_lib.BlockCipherMode = Base.extend({
                    /**
                     * Creates this mode for encryption.
                     *
                     * @param {Cipher} cipher A block cipher instance.
                     * @param {Array} iv The IV words.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var mode = CryptoJS.mode.CBC.createEncryptor(cipher, iv.words);
                     */
                    createEncryptor: function (cipher, iv) {
                        return this.Encryptor.create(cipher, iv);
                    },

                    /**
                     * Creates this mode for decryption.
                     *
                     * @param {Cipher} cipher A block cipher instance.
                     * @param {Array} iv The IV words.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var mode = CryptoJS.mode.CBC.createDecryptor(cipher, iv.words);
                     */
                    createDecryptor: function (cipher, iv) {
                        return this.Decryptor.create(cipher, iv);
                    },

                    /**
                     * Initializes a newly created mode.
                     *
                     * @param {Cipher} cipher A block cipher instance.
                     * @param {Array} iv The IV words.
                     *
                     * @example
                     *
                     *     var mode = CryptoJS.mode.CBC.Encryptor.create(cipher, iv.words);
                     */
                    init: function (cipher, iv) {
                        this._cipher = cipher;
                        this._iv = iv;
                    }
                });

                /**
                 * Cipher Block Chaining mode.
                 */
                var CBC = C_mode.CBC = (function () {
                    /**
                     * Abstract base CBC mode.
                     */
                    var CBC = BlockCipherMode.extend();

                    /**
                     * CBC encryptor.
                     */
                    CBC.Encryptor = CBC.extend({
                        /**
                         * Processes the data block at offset.
                         *
                         * @param {Array} words The data words to operate on.
                         * @param {number} offset The offset where the block starts.
                         *
                         * @example
                         *
                         *     mode.processBlock(data.words, offset);
                         */
                        processBlock: function (words, offset) {
                            // Shortcuts
                            var cipher = this._cipher;
                            var blockSize = cipher.blockSize;

                            // XOR and encrypt
                            xorBlock.call(this, words, offset, blockSize);
                            cipher.encryptBlock(words, offset);

                            // Remember this block to use with next block
                            this._prevBlock = words.slice(offset, offset + blockSize);
                        }
                    });

                    /**
                     * CBC decryptor.
                     */
                    CBC.Decryptor = CBC.extend({
                        /**
                         * Processes the data block at offset.
                         *
                         * @param {Array} words The data words to operate on.
                         * @param {number} offset The offset where the block starts.
                         *
                         * @example
                         *
                         *     mode.processBlock(data.words, offset);
                         */
                        processBlock: function (words, offset) {
                            // Shortcuts
                            var cipher = this._cipher;
                            var blockSize = cipher.blockSize;

                            // Remember this block to use with next block
                            var thisBlock = words.slice(offset, offset + blockSize);

                            // Decrypt and XOR
                            cipher.decryptBlock(words, offset);
                            xorBlock.call(this, words, offset, blockSize);

                            // This block becomes the previous block
                            this._prevBlock = thisBlock;
                        }
                    });

                    function xorBlock(words, offset, blockSize) {
                        // Shortcut
                        var iv = this._iv;

                        // Choose mixing block
                        if (iv) {
                            var block = iv;

                            // Remove IV for subsequent blocks
                            this._iv = undefined;
                        } else {
                            var block = this._prevBlock;
                        }

                        // XOR blocks
                        for (var i = 0; i < blockSize; i++) {
                            words[offset + i] ^= block[i];
                        }
                    }

                    return CBC;
                }());

                /**
                 * Padding namespace.
                 */
                var C_pad = C.pad = {};

                /**
                 * PKCS #5/7 padding strategy.
                 */
                var Pkcs7 = C_pad.Pkcs7 = {
                    /**
                     * Pads data using the algorithm defined in PKCS #5/7.
                     *
                     * @param {WordArray} data The data to pad.
                     * @param {number} blockSize The multiple that the data should be padded to.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     CryptoJS.pad.Pkcs7.pad(wordArray, 4);
                     */
                    pad: function (data, blockSize) {
                        // Shortcut
                        var blockSizeBytes = blockSize * 4;

                        // Count padding bytes
                        var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;

                        // Create padding word
                        var paddingWord = (nPaddingBytes << 24) | (nPaddingBytes << 16) | (nPaddingBytes << 8) | nPaddingBytes;

                        // Create padding
                        var paddingWords = [];
                        for (var i = 0; i < nPaddingBytes; i += 4) {
                            paddingWords.push(paddingWord);
                        }
                        var padding = WordArray.create(paddingWords, nPaddingBytes);

                        // Add padding
                        data.concat(padding);
                    },

                    /**
                     * Unpads data that had been padded using the algorithm defined in PKCS #5/7.
                     *
                     * @param {WordArray} data The data to unpad.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     CryptoJS.pad.Pkcs7.unpad(wordArray);
                     */
                    unpad: function (data) {
                        // Get number of padding bytes from last byte
                        var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;

                        // Remove padding
                        data.sigBytes -= nPaddingBytes;
                    }
                };

                /**
                 * Abstract base block cipher template.
                 *
                 * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 4 (128 bits)
                 */
                var BlockCipher = C_lib.BlockCipher = Cipher.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {Mode} mode The block mode to use. Default: CBC
                     * @property {Padding} padding The padding strategy to use. Default: Pkcs7
                     */
                    cfg: Cipher.cfg.extend({
                        mode: CBC,
                        padding: Pkcs7
                    }),

                    reset: function () {
                        // Reset cipher
                        Cipher.reset.call(this);

                        // Shortcuts
                        var cfg = this.cfg;
                        var iv = cfg.iv;
                        var mode = cfg.mode;

                        // Reset block mode
                        if (this._xformMode == this._ENC_XFORM_MODE) {
                            var modeCreator = mode.createEncryptor;
                        } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
                            var modeCreator = mode.createDecryptor;
                            // Keep at least one block in the buffer for unpadding
                            this._minBufferSize = 1;
                        }

                        if (this._mode && this._mode.__creator == modeCreator) {
                            this._mode.init(this, iv && iv.words);
                        } else {
                            this._mode = modeCreator.call(mode, this, iv && iv.words);
                            this._mode.__creator = modeCreator;
                        }
                    },

                    _doProcessBlock: function (words, offset) {
                        this._mode.processBlock(words, offset);
                    },

                    _doFinalize: function () {
                        // Shortcut
                        var padding = this.cfg.padding;

                        // Finalize
                        if (this._xformMode == this._ENC_XFORM_MODE) {
                            // Pad data
                            padding.pad(this._data, this.blockSize);

                            // Process final blocks
                            var finalProcessedBlocks = this._process(!!'flush');
                        } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
                            // Process final blocks
                            var finalProcessedBlocks = this._process(!!'flush');

                            // Unpad data
                            padding.unpad(finalProcessedBlocks);
                        }

                        return finalProcessedBlocks;
                    },

                    blockSize: 128 / 32
                });

                /**
                 * A collection of cipher parameters.
                 *
                 * @property {WordArray} ciphertext The raw ciphertext.
                 * @property {WordArray} key The key to this ciphertext.
                 * @property {WordArray} iv The IV used in the ciphering operation.
                 * @property {WordArray} salt The salt used with a key derivation function.
                 * @property {Cipher} algorithm The cipher algorithm.
                 * @property {Mode} mode The block mode used in the ciphering operation.
                 * @property {Padding} padding The padding scheme used in the ciphering operation.
                 * @property {number} blockSize The block size of the cipher.
                 * @property {Format} formatter The default formatting strategy to convert this cipher params object to a string.
                 */
                var CipherParams = C_lib.CipherParams = Base.extend({
                    /**
                     * Initializes a newly created cipher params object.
                     *
                     * @param {Object} cipherParams An object with any of the possible cipher parameters.
                     *
                     * @example
                     *
                     *     var cipherParams = CryptoJS.lib.CipherParams.create({
                     *         ciphertext: ciphertextWordArray,
                     *         key: keyWordArray,
                     *         iv: ivWordArray,
                     *         salt: saltWordArray,
                     *         algorithm: CryptoJS.algo.AES,
                     *         mode: CryptoJS.mode.CBC,
                     *         padding: CryptoJS.pad.PKCS7,
                     *         blockSize: 4,
                     *         formatter: CryptoJS.format.OpenSSL
                     *     });
                     */
                    init: function (cipherParams) {
                        this.mixIn(cipherParams);
                    },

                    /**
                     * Converts this cipher params object to a string.
                     *
                     * @param {Format} formatter (Optional) The formatting strategy to use.
                     *
                     * @return {string} The stringified cipher params.
                     *
                     * @throws Error If neither the formatter nor the default formatter is set.
                     *
                     * @example
                     *
                     *     var string = cipherParams + '';
                     *     var string = cipherParams.toString();
                     *     var string = cipherParams.toString(CryptoJS.format.OpenSSL);
                     */
                    toString: function (formatter) {
                        return (formatter || this.formatter).stringify(this);
                    }
                });

                /**
                 * Format namespace.
                 */
                var C_format = C.format = {};

                /**
                 * OpenSSL formatting strategy.
                 */
                var OpenSSLFormatter = C_format.OpenSSL = {
                    /**
                     * Converts a cipher params object to an OpenSSL-compatible string.
                     *
                     * @param {CipherParams} cipherParams The cipher params object.
                     *
                     * @return {string} The OpenSSL-compatible string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var openSSLString = CryptoJS.format.OpenSSL.stringify(cipherParams);
                     */
                    stringify: function (cipherParams) {
                        // Shortcuts
                        var ciphertext = cipherParams.ciphertext;
                        var salt = cipherParams.salt;

                        // Format
                        if (salt) {
                            var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);
                        } else {
                            var wordArray = ciphertext;
                        }

                        return wordArray.toString(Base64);
                    },

                    /**
                     * Converts an OpenSSL-compatible string to a cipher params object.
                     *
                     * @param {string} openSSLStr The OpenSSL-compatible string.
                     *
                     * @return {CipherParams} The cipher params object.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var cipherParams = CryptoJS.format.OpenSSL.parse(openSSLString);
                     */
                    parse: function (openSSLStr) {
                        // Parse base64
                        var ciphertext = Base64.parse(openSSLStr);

                        // Shortcut
                        var ciphertextWords = ciphertext.words;

                        // Test for salt
                        if (ciphertextWords[0] == 0x53616c74 && ciphertextWords[1] == 0x65645f5f) {
                            // Extract salt
                            var salt = WordArray.create(ciphertextWords.slice(2, 4));

                            // Remove salt from ciphertext
                            ciphertextWords.splice(0, 4);
                            ciphertext.sigBytes -= 16;
                        }

                        return CipherParams.create({ciphertext: ciphertext, salt: salt});
                    }
                };

                /**
                 * A cipher wrapper that returns ciphertext as a serializable cipher params object.
                 */
                var SerializableCipher = C_lib.SerializableCipher = Base.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {Formatter} format The formatting strategy to convert cipher param objects to and from a string. Default: OpenSSL
                     */
                    cfg: Base.extend({
                        format: OpenSSLFormatter
                    }),

                    /**
                     * Encrypts a message.
                     *
                     * @param {Cipher} cipher The cipher algorithm to use.
                     * @param {WordArray|string} message The message to encrypt.
                     * @param {WordArray} key The key.
                     * @param {Object} cfg (Optional) The configuration options to use for this operation.
                     *
                     * @return {CipherParams} A cipher params object.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key);
                     *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv });
                     *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv, format: CryptoJS.format.OpenSSL });
                     */
                    encrypt: function (cipher, message, key, cfg) {
                        // Apply config defaults
                        cfg = this.cfg.extend(cfg);

                        // Encrypt
                        var encryptor = cipher.createEncryptor(key, cfg);
                        var ciphertext = encryptor.finalize(message);

                        // Shortcut
                        var cipherCfg = encryptor.cfg;

                        // Create and return serializable cipher params
                        return CipherParams.create({
                            ciphertext: ciphertext,
                            key: key,
                            iv: cipherCfg.iv,
                            algorithm: cipher,
                            mode: cipherCfg.mode,
                            padding: cipherCfg.padding,
                            blockSize: cipher.blockSize,
                            formatter: cfg.format
                        });
                    },

                    /**
                     * Decrypts serialized ciphertext.
                     *
                     * @param {Cipher} cipher The cipher algorithm to use.
                     * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
                     * @param {WordArray} key The key.
                     * @param {Object} cfg (Optional) The configuration options to use for this operation.
                     *
                     * @return {WordArray} The plaintext.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, key, { iv: iv, format: CryptoJS.format.OpenSSL });
                     *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, key, { iv: iv, format: CryptoJS.format.OpenSSL });
                     */
                    decrypt: function (cipher, ciphertext, key, cfg) {
                        // Apply config defaults
                        cfg = this.cfg.extend(cfg);

                        // Convert string to CipherParams
                        ciphertext = this._parse(ciphertext, cfg.format);

                        // Decrypt
                        var plaintext = cipher.createDecryptor(key, cfg).finalize(ciphertext.ciphertext);

                        return plaintext;
                    },

                    /**
                     * Converts serialized ciphertext to CipherParams,
                     * else assumed CipherParams already and returns ciphertext unchanged.
                     *
                     * @param {CipherParams|string} ciphertext The ciphertext.
                     * @param {Formatter} format The formatting strategy to use to parse serialized ciphertext.
                     *
                     * @return {CipherParams} The unserialized ciphertext.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var ciphertextParams = CryptoJS.lib.SerializableCipher._parse(ciphertextStringOrParams, format);
                     */
                    _parse: function (ciphertext, format) {
                        if (typeof ciphertext == 'string') {
                            return format.parse(ciphertext, this);
                        } else {
                            return ciphertext;
                        }
                    }
                });

                /**
                 * Key derivation function namespace.
                 */
                var C_kdf = C.kdf = {};

                /**
                 * OpenSSL key derivation function.
                 */
                var OpenSSLKdf = C_kdf.OpenSSL = {
                    /**
                     * Derives a key and IV from a password.
                     *
                     * @param {string} password The password to derive from.
                     * @param {number} keySize The size in words of the key to generate.
                     * @param {number} ivSize The size in words of the IV to generate.
                     * @param {WordArray|string} salt (Optional) A 64-bit salt to use. If omitted, a salt will be generated randomly.
                     *
                     * @return {CipherParams} A cipher params object with the key, IV, and salt.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32);
                     *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32, 'saltsalt');
                     */
                    execute: function (password, keySize, ivSize, salt) {
                        // Generate random salt
                        if (!salt) {
                            salt = WordArray.random(64 / 8);
                        }

                        // Derive key and IV
                        var key = EvpKDF.create({keySize: keySize + ivSize}).compute(password, salt);

                        // Separate key and IV
                        var iv = WordArray.create(key.words.slice(keySize), ivSize * 4);
                        key.sigBytes = keySize * 4;

                        // Return params
                        return CipherParams.create({key: key, iv: iv, salt: salt});
                    }
                };

                /**
                 * A serializable cipher wrapper that derives the key from a password,
                 * and returns ciphertext as a serializable cipher params object.
                 */
                var PasswordBasedCipher = C_lib.PasswordBasedCipher = SerializableCipher.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {KDF} kdf The key derivation function to use to generate a key and IV from a password. Default: OpenSSL
                     */
                    cfg: SerializableCipher.cfg.extend({
                        kdf: OpenSSLKdf
                    }),

                    /**
                     * Encrypts a message using a password.
                     *
                     * @param {Cipher} cipher The cipher algorithm to use.
                     * @param {WordArray|string} message The message to encrypt.
                     * @param {string} password The password.
                     * @param {Object} cfg (Optional) The configuration options to use for this operation.
                     *
                     * @return {CipherParams} A cipher params object.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password');
                     *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password', { format: CryptoJS.format.OpenSSL });
                     */
                    encrypt: function (cipher, message, password, cfg) {
                        // Apply config defaults
                        cfg = this.cfg.extend(cfg);

                        // Derive key and other params
                        var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize);

                        // Add IV to config
                        cfg.iv = derivedParams.iv;

                        // Encrypt
                        var ciphertext = SerializableCipher.encrypt.call(this, cipher, message, derivedParams.key, cfg);

                        // Mix in derived params
                        ciphertext.mixIn(derivedParams);

                        return ciphertext;
                    },

                    /**
                     * Decrypts serialized ciphertext using a password.
                     *
                     * @param {Cipher} cipher The cipher algorithm to use.
                     * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
                     * @param {string} password The password.
                     * @param {Object} cfg (Optional) The configuration options to use for this operation.
                     *
                     * @return {WordArray} The plaintext.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, 'password', { format: CryptoJS.format.OpenSSL });
                     *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, 'password', { format: CryptoJS.format.OpenSSL });
                     */
                    decrypt: function (cipher, ciphertext, password, cfg) {
                        // Apply config defaults
                        cfg = this.cfg.extend(cfg);

                        // Convert string to CipherParams
                        ciphertext = this._parse(ciphertext, cfg.format);

                        // Derive key and other params
                        var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize, ciphertext.salt);

                        // Add IV to config
                        cfg.iv = derivedParams.iv;

                        // Decrypt
                        var plaintext = SerializableCipher.decrypt.call(this, cipher, ciphertext, derivedParams.key, cfg);

                        return plaintext;
                    }
                });
            }());


        }));

        /***/
    },

    /***/ 229:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * Cipher Feedback block mode.
             */
            CryptoJS.mode.CFB = (function () {
                var CFB = CryptoJS.lib.BlockCipherMode.extend();

                CFB.Encryptor = CFB.extend({
                    processBlock: function (words, offset) {
                        // Shortcuts
                        var cipher = this._cipher;
                        var blockSize = cipher.blockSize;

                        generateKeystreamAndEncrypt.call(this, words, offset, blockSize, cipher);

                        // Remember this block to use with next block
                        this._prevBlock = words.slice(offset, offset + blockSize);
                    }
                });

                CFB.Decryptor = CFB.extend({
                    processBlock: function (words, offset) {
                        // Shortcuts
                        var cipher = this._cipher;
                        var blockSize = cipher.blockSize;

                        // Remember this block to use with next block
                        var thisBlock = words.slice(offset, offset + blockSize);

                        generateKeystreamAndEncrypt.call(this, words, offset, blockSize, cipher);

                        // This block becomes the previous block
                        this._prevBlock = thisBlock;
                    }
                });

                function generateKeystreamAndEncrypt(words, offset, blockSize, cipher) {
                    // Shortcut
                    var iv = this._iv;

                    // Generate keystream
                    if (iv) {
                        var keystream = iv.slice(0);

                        // Remove IV for subsequent blocks
                        this._iv = undefined;
                    } else {
                        var keystream = this._prevBlock;
                    }
                    cipher.encryptBlock(keystream, 0);

                    // Encrypt
                    for (var i = 0; i < blockSize; i++) {
                        words[offset + i] ^= keystream[i];
                    }
                }

                return CFB;
            }());


            return CryptoJS.mode.CFB;

        }));

        /***/
    },

    /***/ 230:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * Counter block mode.
             */
            CryptoJS.mode.CTR = (function () {
                var CTR = CryptoJS.lib.BlockCipherMode.extend();

                var Encryptor = CTR.Encryptor = CTR.extend({
                    processBlock: function (words, offset) {
                        // Shortcuts
                        var cipher = this._cipher
                        var blockSize = cipher.blockSize;
                        var iv = this._iv;
                        var counter = this._counter;

                        // Generate keystream
                        if (iv) {
                            counter = this._counter = iv.slice(0);

                            // Remove IV for subsequent blocks
                            this._iv = undefined;
                        }
                        var keystream = counter.slice(0);
                        cipher.encryptBlock(keystream, 0);

                        // Increment counter
                        counter[blockSize - 1] = (counter[blockSize - 1] + 1) | 0

                        // Encrypt
                        for (var i = 0; i < blockSize; i++) {
                            words[offset + i] ^= keystream[i];
                        }
                    }
                });

                CTR.Decryptor = Encryptor;

                return CTR;
            }());


            return CryptoJS.mode.CTR;

        }));

        /***/
    },

    /***/ 231:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /** @preserve
             * Counter block mode compatible with  Dr Brian Gladman fileenc.c
             * derived from CryptoJS.mode.CTR
             * Jan Hruby jhruby.web@gmail.com
             */
            CryptoJS.mode.CTRGladman = (function () {
                var CTRGladman = CryptoJS.lib.BlockCipherMode.extend();

                function incWord(word) {
                    if (((word >> 24) & 0xff) === 0xff) { //overflow
                        var b1 = (word >> 16) & 0xff;
                        var b2 = (word >> 8) & 0xff;
                        var b3 = word & 0xff;

                        if (b1 === 0xff) // overflow b1
                        {
                            b1 = 0;
                            if (b2 === 0xff) {
                                b2 = 0;
                                if (b3 === 0xff) {
                                    b3 = 0;
                                }
                                else {
                                    ++b3;
                                }
                            }
                            else {
                                ++b2;
                            }
                        }
                        else {
                            ++b1;
                        }

                        word = 0;
                        word += (b1 << 16);
                        word += (b2 << 8);
                        word += b3;
                    }
                    else {
                        word += (0x01 << 24);
                    }
                    return word;
                }

                function incCounter(counter) {
                    if ((counter[0] = incWord(counter[0])) === 0) {
                        // encr_data in fileenc.c from  Dr Brian Gladman's counts only with DWORD j < 8
                        counter[1] = incWord(counter[1]);
                    }
                    return counter;
                }

                var Encryptor = CTRGladman.Encryptor = CTRGladman.extend({
                    processBlock: function (words, offset) {
                        // Shortcuts
                        var cipher = this._cipher
                        var blockSize = cipher.blockSize;
                        var iv = this._iv;
                        var counter = this._counter;

                        // Generate keystream
                        if (iv) {
                            counter = this._counter = iv.slice(0);

                            // Remove IV for subsequent blocks
                            this._iv = undefined;
                        }

                        incCounter(counter);

                        var keystream = counter.slice(0);
                        cipher.encryptBlock(keystream, 0);

                        // Encrypt
                        for (var i = 0; i < blockSize; i++) {
                            words[offset + i] ^= keystream[i];
                        }
                    }
                });

                CTRGladman.Decryptor = Encryptor;

                return CTRGladman;
            }());


            return CryptoJS.mode.CTRGladman;

        }));

        /***/
    },

    /***/ 232:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * Output Feedback block mode.
             */
            CryptoJS.mode.OFB = (function () {
                var OFB = CryptoJS.lib.BlockCipherMode.extend();

                var Encryptor = OFB.Encryptor = OFB.extend({
                    processBlock: function (words, offset) {
                        // Shortcuts
                        var cipher = this._cipher
                        var blockSize = cipher.blockSize;
                        var iv = this._iv;
                        var keystream = this._keystream;

                        // Generate keystream
                        if (iv) {
                            keystream = this._keystream = iv.slice(0);

                            // Remove IV for subsequent blocks
                            this._iv = undefined;
                        }
                        cipher.encryptBlock(keystream, 0);

                        // Encrypt
                        for (var i = 0; i < blockSize; i++) {
                            words[offset + i] ^= keystream[i];
                        }
                    }
                });

                OFB.Decryptor = Encryptor;

                return OFB;
            }());


            return CryptoJS.mode.OFB;

        }));

        /***/
    },

    /***/ 233:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * Electronic Codebook block mode.
             */
            CryptoJS.mode.ECB = (function () {
                var ECB = CryptoJS.lib.BlockCipherMode.extend();

                ECB.Encryptor = ECB.extend({
                    processBlock: function (words, offset) {
                        this._cipher.encryptBlock(words, offset);
                    }
                });

                ECB.Decryptor = ECB.extend({
                    processBlock: function (words, offset) {
                        this._cipher.decryptBlock(words, offset);
                    }
                });

                return ECB;
            }());


            return CryptoJS.mode.ECB;

        }));

        /***/
    },

    /***/ 234:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * ANSI X.923 padding strategy.
             */
            CryptoJS.pad.AnsiX923 = {
                pad: function (data, blockSize) {
                    // Shortcuts
                    var dataSigBytes = data.sigBytes;
                    var blockSizeBytes = blockSize * 4;

                    // Count padding bytes
                    var nPaddingBytes = blockSizeBytes - dataSigBytes % blockSizeBytes;

                    // Compute last byte position
                    var lastBytePos = dataSigBytes + nPaddingBytes - 1;

                    // Pad
                    data.clamp();
                    data.words[lastBytePos >>> 2] |= nPaddingBytes << (24 - (lastBytePos % 4) * 8);
                    data.sigBytes += nPaddingBytes;
                },

                unpad: function (data) {
                    // Get number of padding bytes from last byte
                    var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;

                    // Remove padding
                    data.sigBytes -= nPaddingBytes;
                }
            };


            return CryptoJS.pad.Ansix923;

        }));

        /***/
    },

    /***/ 235:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * ISO 10126 padding strategy.
             */
            CryptoJS.pad.Iso10126 = {
                pad: function (data, blockSize) {
                    // Shortcut
                    var blockSizeBytes = blockSize * 4;

                    // Count padding bytes
                    var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;

                    // Pad
                    data.concat(CryptoJS.lib.WordArray.random(nPaddingBytes - 1)).concat(CryptoJS.lib.WordArray.create([nPaddingBytes << 24], 1));
                },

                unpad: function (data) {
                    // Get number of padding bytes from last byte
                    var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;

                    // Remove padding
                    data.sigBytes -= nPaddingBytes;
                }
            };


            return CryptoJS.pad.Iso10126;

        }));

        /***/
    },

    /***/ 236:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * ISO/IEC 9797-1 Padding Method 2.
             */
            CryptoJS.pad.Iso97971 = {
                pad: function (data, blockSize) {
                    // Add 0x80 byte
                    data.concat(CryptoJS.lib.WordArray.create([0x80000000], 1));

                    // Zero pad the rest
                    CryptoJS.pad.ZeroPadding.pad(data, blockSize);
                },

                unpad: function (data) {
                    // Remove zero padding
                    CryptoJS.pad.ZeroPadding.unpad(data);

                    // Remove one more byte -- the 0x80 byte
                    data.sigBytes--;
                }
            };


            return CryptoJS.pad.Iso97971;

        }));

        /***/
    },

    /***/ 237:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * Zero padding strategy.
             */
            CryptoJS.pad.ZeroPadding = {
                pad: function (data, blockSize) {
                    // Shortcut
                    var blockSizeBytes = blockSize * 4;

                    // Pad
                    data.clamp();
                    data.sigBytes += blockSizeBytes - ((data.sigBytes % blockSizeBytes) || blockSizeBytes);
                },

                unpad: function (data) {
                    // Shortcut
                    var dataWords = data.words;

                    // Unpad
                    var i = data.sigBytes - 1;
                    while (!((dataWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff)) {
                        i--;
                    }
                    data.sigBytes = i + 1;
                }
            };


            return CryptoJS.pad.ZeroPadding;

        }));

        /***/
    },

    /***/ 238:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            /**
             * A noop padding strategy.
             */
            CryptoJS.pad.NoPadding = {
                pad: function () {
                },

                unpad: function () {
                }
            };


            return CryptoJS.pad.NoPadding;

        }));

        /***/
    },

    /***/ 239:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function (undefined) {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var CipherParams = C_lib.CipherParams;
                var C_enc = C.enc;
                var Hex = C_enc.Hex;
                var C_format = C.format;

                var HexFormatter = C_format.Hex = {
                    /**
                     * Converts the ciphertext of a cipher params object to a hexadecimally encoded string.
                     *
                     * @param {CipherParams} cipherParams The cipher params object.
                     *
                     * @return {string} The hexadecimally encoded string.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var hexString = CryptoJS.format.Hex.stringify(cipherParams);
                     */
                    stringify: function (cipherParams) {
                        return cipherParams.ciphertext.toString(Hex);
                    },

                    /**
                     * Converts a hexadecimally encoded ciphertext string to a cipher params object.
                     *
                     * @param {string} input The hexadecimally encoded string.
                     *
                     * @return {CipherParams} The cipher params object.
                     *
                     * @static
                     *
                     * @example
                     *
                     *     var cipherParams = CryptoJS.format.Hex.parse(hexString);
                     */
                    parse: function (input) {
                        var ciphertext = Hex.parse(input);
                        return CipherParams.create({ciphertext: ciphertext});
                    }
                };
            }());


            return CryptoJS.format.Hex;

        }));

        /***/
    },

    /***/ 240:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(216), __webpack_require__(217), __webpack_require__(227), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./enc-base64", "./md5", "./evpkdf", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var BlockCipher = C_lib.BlockCipher;
                var C_algo = C.algo;

                // Lookup tables
                var SBOX = [];
                var INV_SBOX = [];
                var SUB_MIX_0 = [];
                var SUB_MIX_1 = [];
                var SUB_MIX_2 = [];
                var SUB_MIX_3 = [];
                var INV_SUB_MIX_0 = [];
                var INV_SUB_MIX_1 = [];
                var INV_SUB_MIX_2 = [];
                var INV_SUB_MIX_3 = [];

                // Compute lookup tables
                (function () {
                    // Compute double table
                    var d = [];
                    for (var i = 0; i < 256; i++) {
                        if (i < 128) {
                            d[i] = i << 1;
                        } else {
                            d[i] = (i << 1) ^ 0x11b;
                        }
                    }

                    // Walk GF(2^8)
                    var x = 0;
                    var xi = 0;
                    for (var i = 0; i < 256; i++) {
                        // Compute sbox
                        var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4);
                        sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63;
                        SBOX[x] = sx;
                        INV_SBOX[sx] = x;

                        // Compute multiplication
                        var x2 = d[x];
                        var x4 = d[x2];
                        var x8 = d[x4];

                        // Compute sub bytes, mix columns tables
                        var t = (d[sx] * 0x101) ^ (sx * 0x1010100);
                        SUB_MIX_0[x] = (t << 24) | (t >>> 8);
                        SUB_MIX_1[x] = (t << 16) | (t >>> 16);
                        SUB_MIX_2[x] = (t << 8) | (t >>> 24);
                        SUB_MIX_3[x] = t;

                        // Compute inv sub bytes, inv mix columns tables
                        var t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100);
                        INV_SUB_MIX_0[sx] = (t << 24) | (t >>> 8);
                        INV_SUB_MIX_1[sx] = (t << 16) | (t >>> 16);
                        INV_SUB_MIX_2[sx] = (t << 8) | (t >>> 24);
                        INV_SUB_MIX_3[sx] = t;

                        // Compute next counter
                        if (!x) {
                            x = xi = 1;
                        } else {
                            x = x2 ^ d[d[d[x8 ^ x2]]];
                            xi ^= d[d[xi]];
                        }
                    }
                }());

                // Precomputed Rcon lookup
                var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36];

                /**
                 * AES block cipher algorithm.
                 */
                var AES = C_algo.AES = BlockCipher.extend({
                    _doReset: function () {
                        // Skip reset of nRounds has been set before and key did not change
                        if (this._nRounds && this._keyPriorReset === this._key) {
                            return;
                        }

                        // Shortcuts
                        var key = this._keyPriorReset = this._key;
                        var keyWords = key.words;
                        var keySize = key.sigBytes / 4;

                        // Compute number of rounds
                        var nRounds = this._nRounds = keySize + 6;

                        // Compute number of key schedule rows
                        var ksRows = (nRounds + 1) * 4;

                        // Compute key schedule
                        var keySchedule = this._keySchedule = [];
                        for (var ksRow = 0; ksRow < ksRows; ksRow++) {
                            if (ksRow < keySize) {
                                keySchedule[ksRow] = keyWords[ksRow];
                            } else {
                                var t = keySchedule[ksRow - 1];

                                if (!(ksRow % keySize)) {
                                    // Rot word
                                    t = (t << 8) | (t >>> 24);

                                    // Sub word
                                    t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];

                                    // Mix Rcon
                                    t ^= RCON[(ksRow / keySize) | 0] << 24;
                                } else if (keySize > 6 && ksRow % keySize == 4) {
                                    // Sub word
                                    t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];
                                }

                                keySchedule[ksRow] = keySchedule[ksRow - keySize] ^ t;
                            }
                        }

                        // Compute inv key schedule
                        var invKeySchedule = this._invKeySchedule = [];
                        for (var invKsRow = 0; invKsRow < ksRows; invKsRow++) {
                            var ksRow = ksRows - invKsRow;

                            if (invKsRow % 4) {
                                var t = keySchedule[ksRow];
                            } else {
                                var t = keySchedule[ksRow - 4];
                            }

                            if (invKsRow < 4 || ksRow <= 4) {
                                invKeySchedule[invKsRow] = t;
                            } else {
                                invKeySchedule[invKsRow] = INV_SUB_MIX_0[SBOX[t >>> 24]] ^ INV_SUB_MIX_1[SBOX[(t >>> 16) & 0xff]] ^
                                    INV_SUB_MIX_2[SBOX[(t >>> 8) & 0xff]] ^ INV_SUB_MIX_3[SBOX[t & 0xff]];
                            }
                        }
                    },

                    encryptBlock: function (M, offset) {
                        this._doCryptBlock(M, offset, this._keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX);
                    },

                    decryptBlock: function (M, offset) {
                        // Swap 2nd and 4th rows
                        var t = M[offset + 1];
                        M[offset + 1] = M[offset + 3];
                        M[offset + 3] = t;

                        this._doCryptBlock(M, offset, this._invKeySchedule, INV_SUB_MIX_0, INV_SUB_MIX_1, INV_SUB_MIX_2, INV_SUB_MIX_3, INV_SBOX);

                        // Inv swap 2nd and 4th rows
                        var t = M[offset + 1];
                        M[offset + 1] = M[offset + 3];
                        M[offset + 3] = t;
                    },

                    _doCryptBlock: function (M, offset, keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX) {
                        // Shortcut
                        var nRounds = this._nRounds;

                        // Get input, add round key
                        var s0 = M[offset] ^ keySchedule[0];
                        var s1 = M[offset + 1] ^ keySchedule[1];
                        var s2 = M[offset + 2] ^ keySchedule[2];
                        var s3 = M[offset + 3] ^ keySchedule[3];

                        // Key schedule row counter
                        var ksRow = 4;

                        // Rounds
                        for (var round = 1; round < nRounds; round++) {
                            // Shift rows, sub bytes, mix columns, add round key
                            var t0 = SUB_MIX_0[s0 >>> 24] ^ SUB_MIX_1[(s1 >>> 16) & 0xff] ^ SUB_MIX_2[(s2 >>> 8) & 0xff] ^ SUB_MIX_3[s3 & 0xff] ^ keySchedule[ksRow++];
                            var t1 = SUB_MIX_0[s1 >>> 24] ^ SUB_MIX_1[(s2 >>> 16) & 0xff] ^ SUB_MIX_2[(s3 >>> 8) & 0xff] ^ SUB_MIX_3[s0 & 0xff] ^ keySchedule[ksRow++];
                            var t2 = SUB_MIX_0[s2 >>> 24] ^ SUB_MIX_1[(s3 >>> 16) & 0xff] ^ SUB_MIX_2[(s0 >>> 8) & 0xff] ^ SUB_MIX_3[s1 & 0xff] ^ keySchedule[ksRow++];
                            var t3 = SUB_MIX_0[s3 >>> 24] ^ SUB_MIX_1[(s0 >>> 16) & 0xff] ^ SUB_MIX_2[(s1 >>> 8) & 0xff] ^ SUB_MIX_3[s2 & 0xff] ^ keySchedule[ksRow++];

                            // Update state
                            s0 = t0;
                            s1 = t1;
                            s2 = t2;
                            s3 = t3;
                        }

                        // Shift rows, sub bytes, add round key
                        var t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++];
                        var t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++];
                        var t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++];
                        var t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++];

                        // Set output
                        M[offset] = t0;
                        M[offset + 1] = t1;
                        M[offset + 2] = t2;
                        M[offset + 3] = t3;
                    },

                    keySize: 256 / 32
                });

                /**
                 * Shortcut functions to the cipher's object interface.
                 *
                 * @example
                 *
                 *     var ciphertext = CryptoJS.AES.encrypt(message, key, cfg);
                 *     var plaintext  = CryptoJS.AES.decrypt(ciphertext, key, cfg);
                 */
                C.AES = BlockCipher._createHelper(AES);
            }());


            return CryptoJS.AES;

        }));

        /***/
    },

    /***/ 241:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(216), __webpack_require__(217), __webpack_require__(227), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./enc-base64", "./md5", "./evpkdf", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var WordArray = C_lib.WordArray;
                var BlockCipher = C_lib.BlockCipher;
                var C_algo = C.algo;

                // Permuted Choice 1 constants
                var PC1 = [
                    57, 49, 41, 33, 25, 17, 9, 1,
                    58, 50, 42, 34, 26, 18, 10, 2,
                    59, 51, 43, 35, 27, 19, 11, 3,
                    60, 52, 44, 36, 63, 55, 47, 39,
                    31, 23, 15, 7, 62, 54, 46, 38,
                    30, 22, 14, 6, 61, 53, 45, 37,
                    29, 21, 13, 5, 28, 20, 12, 4
                ];

                // Permuted Choice 2 constants
                var PC2 = [
                    14, 17, 11, 24, 1, 5,
                    3, 28, 15, 6, 21, 10,
                    23, 19, 12, 4, 26, 8,
                    16, 7, 27, 20, 13, 2,
                    41, 52, 31, 37, 47, 55,
                    30, 40, 51, 45, 33, 48,
                    44, 49, 39, 56, 34, 53,
                    46, 42, 50, 36, 29, 32
                ];

                // Cumulative bit shift constants
                var BIT_SHIFTS = [1, 2, 4, 6, 8, 10, 12, 14, 15, 17, 19, 21, 23, 25, 27, 28];

                // SBOXes and round permutation constants
                var SBOX_P = [
                    {
                        0x0: 0x808200,
                        0x10000000: 0x8000,
                        0x20000000: 0x808002,
                        0x30000000: 0x2,
                        0x40000000: 0x200,
                        0x50000000: 0x808202,
                        0x60000000: 0x800202,
                        0x70000000: 0x800000,
                        0x80000000: 0x202,
                        0x90000000: 0x800200,
                        0xa0000000: 0x8200,
                        0xb0000000: 0x808000,
                        0xc0000000: 0x8002,
                        0xd0000000: 0x800002,
                        0xe0000000: 0x0,
                        0xf0000000: 0x8202,
                        0x8000000: 0x0,
                        0x18000000: 0x808202,
                        0x28000000: 0x8202,
                        0x38000000: 0x8000,
                        0x48000000: 0x808200,
                        0x58000000: 0x200,
                        0x68000000: 0x808002,
                        0x78000000: 0x2,
                        0x88000000: 0x800200,
                        0x98000000: 0x8200,
                        0xa8000000: 0x808000,
                        0xb8000000: 0x800202,
                        0xc8000000: 0x800002,
                        0xd8000000: 0x8002,
                        0xe8000000: 0x202,
                        0xf8000000: 0x800000,
                        0x1: 0x8000,
                        0x10000001: 0x2,
                        0x20000001: 0x808200,
                        0x30000001: 0x800000,
                        0x40000001: 0x808002,
                        0x50000001: 0x8200,
                        0x60000001: 0x200,
                        0x70000001: 0x800202,
                        0x80000001: 0x808202,
                        0x90000001: 0x808000,
                        0xa0000001: 0x800002,
                        0xb0000001: 0x8202,
                        0xc0000001: 0x202,
                        0xd0000001: 0x800200,
                        0xe0000001: 0x8002,
                        0xf0000001: 0x0,
                        0x8000001: 0x808202,
                        0x18000001: 0x808000,
                        0x28000001: 0x800000,
                        0x38000001: 0x200,
                        0x48000001: 0x8000,
                        0x58000001: 0x800002,
                        0x68000001: 0x2,
                        0x78000001: 0x8202,
                        0x88000001: 0x8002,
                        0x98000001: 0x800202,
                        0xa8000001: 0x202,
                        0xb8000001: 0x808200,
                        0xc8000001: 0x800200,
                        0xd8000001: 0x0,
                        0xe8000001: 0x8200,
                        0xf8000001: 0x808002
                    },
                    {
                        0x0: 0x40084010,
                        0x1000000: 0x4000,
                        0x2000000: 0x80000,
                        0x3000000: 0x40080010,
                        0x4000000: 0x40000010,
                        0x5000000: 0x40084000,
                        0x6000000: 0x40004000,
                        0x7000000: 0x10,
                        0x8000000: 0x84000,
                        0x9000000: 0x40004010,
                        0xa000000: 0x40000000,
                        0xb000000: 0x84010,
                        0xc000000: 0x80010,
                        0xd000000: 0x0,
                        0xe000000: 0x4010,
                        0xf000000: 0x40080000,
                        0x800000: 0x40004000,
                        0x1800000: 0x84010,
                        0x2800000: 0x10,
                        0x3800000: 0x40004010,
                        0x4800000: 0x40084010,
                        0x5800000: 0x40000000,
                        0x6800000: 0x80000,
                        0x7800000: 0x40080010,
                        0x8800000: 0x80010,
                        0x9800000: 0x0,
                        0xa800000: 0x4000,
                        0xb800000: 0x40080000,
                        0xc800000: 0x40000010,
                        0xd800000: 0x84000,
                        0xe800000: 0x40084000,
                        0xf800000: 0x4010,
                        0x10000000: 0x0,
                        0x11000000: 0x40080010,
                        0x12000000: 0x40004010,
                        0x13000000: 0x40084000,
                        0x14000000: 0x40080000,
                        0x15000000: 0x10,
                        0x16000000: 0x84010,
                        0x17000000: 0x4000,
                        0x18000000: 0x4010,
                        0x19000000: 0x80000,
                        0x1a000000: 0x80010,
                        0x1b000000: 0x40000010,
                        0x1c000000: 0x84000,
                        0x1d000000: 0x40004000,
                        0x1e000000: 0x40000000,
                        0x1f000000: 0x40084010,
                        0x10800000: 0x84010,
                        0x11800000: 0x80000,
                        0x12800000: 0x40080000,
                        0x13800000: 0x4000,
                        0x14800000: 0x40004000,
                        0x15800000: 0x40084010,
                        0x16800000: 0x10,
                        0x17800000: 0x40000000,
                        0x18800000: 0x40084000,
                        0x19800000: 0x40000010,
                        0x1a800000: 0x40004010,
                        0x1b800000: 0x80010,
                        0x1c800000: 0x0,
                        0x1d800000: 0x4010,
                        0x1e800000: 0x40080010,
                        0x1f800000: 0x84000
                    },
                    {
                        0x0: 0x104,
                        0x100000: 0x0,
                        0x200000: 0x4000100,
                        0x300000: 0x10104,
                        0x400000: 0x10004,
                        0x500000: 0x4000004,
                        0x600000: 0x4010104,
                        0x700000: 0x4010000,
                        0x800000: 0x4000000,
                        0x900000: 0x4010100,
                        0xa00000: 0x10100,
                        0xb00000: 0x4010004,
                        0xc00000: 0x4000104,
                        0xd00000: 0x10000,
                        0xe00000: 0x4,
                        0xf00000: 0x100,
                        0x80000: 0x4010100,
                        0x180000: 0x4010004,
                        0x280000: 0x0,
                        0x380000: 0x4000100,
                        0x480000: 0x4000004,
                        0x580000: 0x10000,
                        0x680000: 0x10004,
                        0x780000: 0x104,
                        0x880000: 0x4,
                        0x980000: 0x100,
                        0xa80000: 0x4010000,
                        0xb80000: 0x10104,
                        0xc80000: 0x10100,
                        0xd80000: 0x4000104,
                        0xe80000: 0x4010104,
                        0xf80000: 0x4000000,
                        0x1000000: 0x4010100,
                        0x1100000: 0x10004,
                        0x1200000: 0x10000,
                        0x1300000: 0x4000100,
                        0x1400000: 0x100,
                        0x1500000: 0x4010104,
                        0x1600000: 0x4000004,
                        0x1700000: 0x0,
                        0x1800000: 0x4000104,
                        0x1900000: 0x4000000,
                        0x1a00000: 0x4,
                        0x1b00000: 0x10100,
                        0x1c00000: 0x4010000,
                        0x1d00000: 0x104,
                        0x1e00000: 0x10104,
                        0x1f00000: 0x4010004,
                        0x1080000: 0x4000000,
                        0x1180000: 0x104,
                        0x1280000: 0x4010100,
                        0x1380000: 0x0,
                        0x1480000: 0x10004,
                        0x1580000: 0x4000100,
                        0x1680000: 0x100,
                        0x1780000: 0x4010004,
                        0x1880000: 0x10000,
                        0x1980000: 0x4010104,
                        0x1a80000: 0x10104,
                        0x1b80000: 0x4000004,
                        0x1c80000: 0x4000104,
                        0x1d80000: 0x4010000,
                        0x1e80000: 0x4,
                        0x1f80000: 0x10100
                    },
                    {
                        0x0: 0x80401000,
                        0x10000: 0x80001040,
                        0x20000: 0x401040,
                        0x30000: 0x80400000,
                        0x40000: 0x0,
                        0x50000: 0x401000,
                        0x60000: 0x80000040,
                        0x70000: 0x400040,
                        0x80000: 0x80000000,
                        0x90000: 0x400000,
                        0xa0000: 0x40,
                        0xb0000: 0x80001000,
                        0xc0000: 0x80400040,
                        0xd0000: 0x1040,
                        0xe0000: 0x1000,
                        0xf0000: 0x80401040,
                        0x8000: 0x80001040,
                        0x18000: 0x40,
                        0x28000: 0x80400040,
                        0x38000: 0x80001000,
                        0x48000: 0x401000,
                        0x58000: 0x80401040,
                        0x68000: 0x0,
                        0x78000: 0x80400000,
                        0x88000: 0x1000,
                        0x98000: 0x80401000,
                        0xa8000: 0x400000,
                        0xb8000: 0x1040,
                        0xc8000: 0x80000000,
                        0xd8000: 0x400040,
                        0xe8000: 0x401040,
                        0xf8000: 0x80000040,
                        0x100000: 0x400040,
                        0x110000: 0x401000,
                        0x120000: 0x80000040,
                        0x130000: 0x0,
                        0x140000: 0x1040,
                        0x150000: 0x80400040,
                        0x160000: 0x80401000,
                        0x170000: 0x80001040,
                        0x180000: 0x80401040,
                        0x190000: 0x80000000,
                        0x1a0000: 0x80400000,
                        0x1b0000: 0x401040,
                        0x1c0000: 0x80001000,
                        0x1d0000: 0x400000,
                        0x1e0000: 0x40,
                        0x1f0000: 0x1000,
                        0x108000: 0x80400000,
                        0x118000: 0x80401040,
                        0x128000: 0x0,
                        0x138000: 0x401000,
                        0x148000: 0x400040,
                        0x158000: 0x80000000,
                        0x168000: 0x80001040,
                        0x178000: 0x40,
                        0x188000: 0x80000040,
                        0x198000: 0x1000,
                        0x1a8000: 0x80001000,
                        0x1b8000: 0x80400040,
                        0x1c8000: 0x1040,
                        0x1d8000: 0x80401000,
                        0x1e8000: 0x400000,
                        0x1f8000: 0x401040
                    },
                    {
                        0x0: 0x80,
                        0x1000: 0x1040000,
                        0x2000: 0x40000,
                        0x3000: 0x20000000,
                        0x4000: 0x20040080,
                        0x5000: 0x1000080,
                        0x6000: 0x21000080,
                        0x7000: 0x40080,
                        0x8000: 0x1000000,
                        0x9000: 0x20040000,
                        0xa000: 0x20000080,
                        0xb000: 0x21040080,
                        0xc000: 0x21040000,
                        0xd000: 0x0,
                        0xe000: 0x1040080,
                        0xf000: 0x21000000,
                        0x800: 0x1040080,
                        0x1800: 0x21000080,
                        0x2800: 0x80,
                        0x3800: 0x1040000,
                        0x4800: 0x40000,
                        0x5800: 0x20040080,
                        0x6800: 0x21040000,
                        0x7800: 0x20000000,
                        0x8800: 0x20040000,
                        0x9800: 0x0,
                        0xa800: 0x21040080,
                        0xb800: 0x1000080,
                        0xc800: 0x20000080,
                        0xd800: 0x21000000,
                        0xe800: 0x1000000,
                        0xf800: 0x40080,
                        0x10000: 0x40000,
                        0x11000: 0x80,
                        0x12000: 0x20000000,
                        0x13000: 0x21000080,
                        0x14000: 0x1000080,
                        0x15000: 0x21040000,
                        0x16000: 0x20040080,
                        0x17000: 0x1000000,
                        0x18000: 0x21040080,
                        0x19000: 0x21000000,
                        0x1a000: 0x1040000,
                        0x1b000: 0x20040000,
                        0x1c000: 0x40080,
                        0x1d000: 0x20000080,
                        0x1e000: 0x0,
                        0x1f000: 0x1040080,
                        0x10800: 0x21000080,
                        0x11800: 0x1000000,
                        0x12800: 0x1040000,
                        0x13800: 0x20040080,
                        0x14800: 0x20000000,
                        0x15800: 0x1040080,
                        0x16800: 0x80,
                        0x17800: 0x21040000,
                        0x18800: 0x40080,
                        0x19800: 0x21040080,
                        0x1a800: 0x0,
                        0x1b800: 0x21000000,
                        0x1c800: 0x1000080,
                        0x1d800: 0x40000,
                        0x1e800: 0x20040000,
                        0x1f800: 0x20000080
                    },
                    {
                        0x0: 0x10000008,
                        0x100: 0x2000,
                        0x200: 0x10200000,
                        0x300: 0x10202008,
                        0x400: 0x10002000,
                        0x500: 0x200000,
                        0x600: 0x200008,
                        0x700: 0x10000000,
                        0x800: 0x0,
                        0x900: 0x10002008,
                        0xa00: 0x202000,
                        0xb00: 0x8,
                        0xc00: 0x10200008,
                        0xd00: 0x202008,
                        0xe00: 0x2008,
                        0xf00: 0x10202000,
                        0x80: 0x10200000,
                        0x180: 0x10202008,
                        0x280: 0x8,
                        0x380: 0x200000,
                        0x480: 0x202008,
                        0x580: 0x10000008,
                        0x680: 0x10002000,
                        0x780: 0x2008,
                        0x880: 0x200008,
                        0x980: 0x2000,
                        0xa80: 0x10002008,
                        0xb80: 0x10200008,
                        0xc80: 0x0,
                        0xd80: 0x10202000,
                        0xe80: 0x202000,
                        0xf80: 0x10000000,
                        0x1000: 0x10002000,
                        0x1100: 0x10200008,
                        0x1200: 0x10202008,
                        0x1300: 0x2008,
                        0x1400: 0x200000,
                        0x1500: 0x10000000,
                        0x1600: 0x10000008,
                        0x1700: 0x202000,
                        0x1800: 0x202008,
                        0x1900: 0x0,
                        0x1a00: 0x8,
                        0x1b00: 0x10200000,
                        0x1c00: 0x2000,
                        0x1d00: 0x10002008,
                        0x1e00: 0x10202000,
                        0x1f00: 0x200008,
                        0x1080: 0x8,
                        0x1180: 0x202000,
                        0x1280: 0x200000,
                        0x1380: 0x10000008,
                        0x1480: 0x10002000,
                        0x1580: 0x2008,
                        0x1680: 0x10202008,
                        0x1780: 0x10200000,
                        0x1880: 0x10202000,
                        0x1980: 0x10200008,
                        0x1a80: 0x2000,
                        0x1b80: 0x202008,
                        0x1c80: 0x200008,
                        0x1d80: 0x0,
                        0x1e80: 0x10000000,
                        0x1f80: 0x10002008
                    },
                    {
                        0x0: 0x100000,
                        0x10: 0x2000401,
                        0x20: 0x400,
                        0x30: 0x100401,
                        0x40: 0x2100401,
                        0x50: 0x0,
                        0x60: 0x1,
                        0x70: 0x2100001,
                        0x80: 0x2000400,
                        0x90: 0x100001,
                        0xa0: 0x2000001,
                        0xb0: 0x2100400,
                        0xc0: 0x2100000,
                        0xd0: 0x401,
                        0xe0: 0x100400,
                        0xf0: 0x2000000,
                        0x8: 0x2100001,
                        0x18: 0x0,
                        0x28: 0x2000401,
                        0x38: 0x2100400,
                        0x48: 0x100000,
                        0x58: 0x2000001,
                        0x68: 0x2000000,
                        0x78: 0x401,
                        0x88: 0x100401,
                        0x98: 0x2000400,
                        0xa8: 0x2100000,
                        0xb8: 0x100001,
                        0xc8: 0x400,
                        0xd8: 0x2100401,
                        0xe8: 0x1,
                        0xf8: 0x100400,
                        0x100: 0x2000000,
                        0x110: 0x100000,
                        0x120: 0x2000401,
                        0x130: 0x2100001,
                        0x140: 0x100001,
                        0x150: 0x2000400,
                        0x160: 0x2100400,
                        0x170: 0x100401,
                        0x180: 0x401,
                        0x190: 0x2100401,
                        0x1a0: 0x100400,
                        0x1b0: 0x1,
                        0x1c0: 0x0,
                        0x1d0: 0x2100000,
                        0x1e0: 0x2000001,
                        0x1f0: 0x400,
                        0x108: 0x100400,
                        0x118: 0x2000401,
                        0x128: 0x2100001,
                        0x138: 0x1,
                        0x148: 0x2000000,
                        0x158: 0x100000,
                        0x168: 0x401,
                        0x178: 0x2100400,
                        0x188: 0x2000001,
                        0x198: 0x2100000,
                        0x1a8: 0x0,
                        0x1b8: 0x2100401,
                        0x1c8: 0x100401,
                        0x1d8: 0x400,
                        0x1e8: 0x2000400,
                        0x1f8: 0x100001
                    },
                    {
                        0x0: 0x8000820,
                        0x1: 0x20000,
                        0x2: 0x8000000,
                        0x3: 0x20,
                        0x4: 0x20020,
                        0x5: 0x8020820,
                        0x6: 0x8020800,
                        0x7: 0x800,
                        0x8: 0x8020000,
                        0x9: 0x8000800,
                        0xa: 0x20800,
                        0xb: 0x8020020,
                        0xc: 0x820,
                        0xd: 0x0,
                        0xe: 0x8000020,
                        0xf: 0x20820,
                        0x80000000: 0x800,
                        0x80000001: 0x8020820,
                        0x80000002: 0x8000820,
                        0x80000003: 0x8000000,
                        0x80000004: 0x8020000,
                        0x80000005: 0x20800,
                        0x80000006: 0x20820,
                        0x80000007: 0x20,
                        0x80000008: 0x8000020,
                        0x80000009: 0x820,
                        0x8000000a: 0x20020,
                        0x8000000b: 0x8020800,
                        0x8000000c: 0x0,
                        0x8000000d: 0x8020020,
                        0x8000000e: 0x8000800,
                        0x8000000f: 0x20000,
                        0x10: 0x20820,
                        0x11: 0x8020800,
                        0x12: 0x20,
                        0x13: 0x800,
                        0x14: 0x8000800,
                        0x15: 0x8000020,
                        0x16: 0x8020020,
                        0x17: 0x20000,
                        0x18: 0x0,
                        0x19: 0x20020,
                        0x1a: 0x8020000,
                        0x1b: 0x8000820,
                        0x1c: 0x8020820,
                        0x1d: 0x20800,
                        0x1e: 0x820,
                        0x1f: 0x8000000,
                        0x80000010: 0x20000,
                        0x80000011: 0x800,
                        0x80000012: 0x8020020,
                        0x80000013: 0x20820,
                        0x80000014: 0x20,
                        0x80000015: 0x8020000,
                        0x80000016: 0x8000000,
                        0x80000017: 0x8000820,
                        0x80000018: 0x8020820,
                        0x80000019: 0x8000020,
                        0x8000001a: 0x8000800,
                        0x8000001b: 0x0,
                        0x8000001c: 0x20800,
                        0x8000001d: 0x820,
                        0x8000001e: 0x20020,
                        0x8000001f: 0x8020800
                    }
                ];

                // Masks that select the SBOX input
                var SBOX_MASK = [
                    0xf8000001, 0x1f800000, 0x01f80000, 0x001f8000,
                    0x0001f800, 0x00001f80, 0x000001f8, 0x8000001f
                ];

                /**
                 * DES block cipher algorithm.
                 */
                var DES = C_algo.DES = BlockCipher.extend({
                    _doReset: function () {
                        // Shortcuts
                        var key = this._key;
                        var keyWords = key.words;

                        // Select 56 bits according to PC1
                        var keyBits = [];
                        for (var i = 0; i < 56; i++) {
                            var keyBitPos = PC1[i] - 1;
                            keyBits[i] = (keyWords[keyBitPos >>> 5] >>> (31 - keyBitPos % 32)) & 1;
                        }

                        // Assemble 16 subkeys
                        var subKeys = this._subKeys = [];
                        for (var nSubKey = 0; nSubKey < 16; nSubKey++) {
                            // Create subkey
                            var subKey = subKeys[nSubKey] = [];

                            // Shortcut
                            var bitShift = BIT_SHIFTS[nSubKey];

                            // Select 48 bits according to PC2
                            for (var i = 0; i < 24; i++) {
                                // Select from the left 28 key bits
                                subKey[(i / 6) | 0] |= keyBits[((PC2[i] - 1) + bitShift) % 28] << (31 - i % 6);

                                // Select from the right 28 key bits
                                subKey[4 + ((i / 6) | 0)] |= keyBits[28 + (((PC2[i + 24] - 1) + bitShift) % 28)] << (31 - i % 6);
                            }

                            // Since each subkey is applied to an expanded 32-bit input,
                            // the subkey can be broken into 8 values scaled to 32-bits,
                            // which allows the key to be used without expansion
                            subKey[0] = (subKey[0] << 1) | (subKey[0] >>> 31);
                            for (var i = 1; i < 7; i++) {
                                subKey[i] = subKey[i] >>> ((i - 1) * 4 + 3);
                            }
                            subKey[7] = (subKey[7] << 5) | (subKey[7] >>> 27);
                        }

                        // Compute inverse subkeys
                        var invSubKeys = this._invSubKeys = [];
                        for (var i = 0; i < 16; i++) {
                            invSubKeys[i] = subKeys[15 - i];
                        }
                    },

                    encryptBlock: function (M, offset) {
                        this._doCryptBlock(M, offset, this._subKeys);
                    },

                    decryptBlock: function (M, offset) {
                        this._doCryptBlock(M, offset, this._invSubKeys);
                    },

                    _doCryptBlock: function (M, offset, subKeys) {
                        // Get input
                        this._lBlock = M[offset];
                        this._rBlock = M[offset + 1];

                        // Initial permutation
                        exchangeLR.call(this, 4, 0x0f0f0f0f);
                        exchangeLR.call(this, 16, 0x0000ffff);
                        exchangeRL.call(this, 2, 0x33333333);
                        exchangeRL.call(this, 8, 0x00ff00ff);
                        exchangeLR.call(this, 1, 0x55555555);

                        // Rounds
                        for (var round = 0; round < 16; round++) {
                            // Shortcuts
                            var subKey = subKeys[round];
                            var lBlock = this._lBlock;
                            var rBlock = this._rBlock;

                            // Feistel function
                            var f = 0;
                            for (var i = 0; i < 8; i++) {
                                f |= SBOX_P[i][((rBlock ^ subKey[i]) & SBOX_MASK[i]) >>> 0];
                            }
                            this._lBlock = rBlock;
                            this._rBlock = lBlock ^ f;
                        }

                        // Undo swap from last round
                        var t = this._lBlock;
                        this._lBlock = this._rBlock;
                        this._rBlock = t;

                        // Final permutation
                        exchangeLR.call(this, 1, 0x55555555);
                        exchangeRL.call(this, 8, 0x00ff00ff);
                        exchangeRL.call(this, 2, 0x33333333);
                        exchangeLR.call(this, 16, 0x0000ffff);
                        exchangeLR.call(this, 4, 0x0f0f0f0f);

                        // Set output
                        M[offset] = this._lBlock;
                        M[offset + 1] = this._rBlock;
                    },

                    keySize: 64 / 32,

                    ivSize: 64 / 32,

                    blockSize: 64 / 32
                });

                // Swap bits across the left and right words
                function exchangeLR(offset, mask) {
                    var t = ((this._lBlock >>> offset) ^ this._rBlock) & mask;
                    this._rBlock ^= t;
                    this._lBlock ^= t << offset;
                }

                function exchangeRL(offset, mask) {
                    var t = ((this._rBlock >>> offset) ^ this._lBlock) & mask;
                    this._lBlock ^= t;
                    this._rBlock ^= t << offset;
                }

                /**
                 * Shortcut functions to the cipher's object interface.
                 *
                 * @example
                 *
                 *     var ciphertext = CryptoJS.DES.encrypt(message, key, cfg);
                 *     var plaintext  = CryptoJS.DES.decrypt(ciphertext, key, cfg);
                 */
                C.DES = BlockCipher._createHelper(DES);

                /**
                 * Triple-DES block cipher algorithm.
                 */
                var TripleDES = C_algo.TripleDES = BlockCipher.extend({
                    _doReset: function () {
                        // Shortcuts
                        var key = this._key;
                        var keyWords = key.words;

                        // Create DES instances
                        this._des1 = DES.createEncryptor(WordArray.create(keyWords.slice(0, 2)));
                        this._des2 = DES.createEncryptor(WordArray.create(keyWords.slice(2, 4)));
                        this._des3 = DES.createEncryptor(WordArray.create(keyWords.slice(4, 6)));
                    },

                    encryptBlock: function (M, offset) {
                        this._des1.encryptBlock(M, offset);
                        this._des2.decryptBlock(M, offset);
                        this._des3.encryptBlock(M, offset);
                    },

                    decryptBlock: function (M, offset) {
                        this._des3.decryptBlock(M, offset);
                        this._des2.encryptBlock(M, offset);
                        this._des1.decryptBlock(M, offset);
                    },

                    keySize: 192 / 32,

                    ivSize: 64 / 32,

                    blockSize: 64 / 32
                });

                /**
                 * Shortcut functions to the cipher's object interface.
                 *
                 * @example
                 *
                 *     var ciphertext = CryptoJS.TripleDES.encrypt(message, key, cfg);
                 *     var plaintext  = CryptoJS.TripleDES.decrypt(ciphertext, key, cfg);
                 */
                C.TripleDES = BlockCipher._createHelper(TripleDES);
            }());


            return CryptoJS.TripleDES;

        }));

        /***/
    },

    /***/ 242:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(216), __webpack_require__(217), __webpack_require__(227), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./enc-base64", "./md5", "./evpkdf", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var StreamCipher = C_lib.StreamCipher;
                var C_algo = C.algo;

                /**
                 * RC4 stream cipher algorithm.
                 */
                var RC4 = C_algo.RC4 = StreamCipher.extend({
                    _doReset: function () {
                        // Shortcuts
                        var key = this._key;
                        var keyWords = key.words;
                        var keySigBytes = key.sigBytes;

                        // Init sbox
                        var S = this._S = [];
                        for (var i = 0; i < 256; i++) {
                            S[i] = i;
                        }

                        // Key setup
                        for (var i = 0, j = 0; i < 256; i++) {
                            var keyByteIndex = i % keySigBytes;
                            var keyByte = (keyWords[keyByteIndex >>> 2] >>> (24 - (keyByteIndex % 4) * 8)) & 0xff;

                            j = (j + S[i] + keyByte) % 256;

                            // Swap
                            var t = S[i];
                            S[i] = S[j];
                            S[j] = t;
                        }

                        // Counters
                        this._i = this._j = 0;
                    },

                    _doProcessBlock: function (M, offset) {
                        M[offset] ^= generateKeystreamWord.call(this);
                    },

                    keySize: 256 / 32,

                    ivSize: 0
                });

                function generateKeystreamWord() {
                    // Shortcuts
                    var S = this._S;
                    var i = this._i;
                    var j = this._j;

                    // Generate keystream word
                    var keystreamWord = 0;
                    for (var n = 0; n < 4; n++) {
                        i = (i + 1) % 256;
                        j = (j + S[i]) % 256;

                        // Swap
                        var t = S[i];
                        S[i] = S[j];
                        S[j] = t;

                        keystreamWord |= S[(S[i] + S[j]) % 256] << (24 - n * 8);
                    }

                    // Update counters
                    this._i = i;
                    this._j = j;

                    return keystreamWord;
                }

                /**
                 * Shortcut functions to the cipher's object interface.
                 *
                 * @example
                 *
                 *     var ciphertext = CryptoJS.RC4.encrypt(message, key, cfg);
                 *     var plaintext  = CryptoJS.RC4.decrypt(ciphertext, key, cfg);
                 */
                C.RC4 = StreamCipher._createHelper(RC4);

                /**
                 * Modified RC4 stream cipher algorithm.
                 */
                var RC4Drop = C_algo.RC4Drop = RC4.extend({
                    /**
                     * Configuration options.
                     *
                     * @property {number} drop The number of keystream words to drop. Default 192
                     */
                    cfg: RC4.cfg.extend({
                        drop: 192
                    }),

                    _doReset: function () {
                        RC4._doReset.call(this);

                        // Drop
                        for (var i = this.cfg.drop; i > 0; i--) {
                            generateKeystreamWord.call(this);
                        }
                    }
                });

                /**
                 * Shortcut functions to the cipher's object interface.
                 *
                 * @example
                 *
                 *     var ciphertext = CryptoJS.RC4Drop.encrypt(message, key, cfg);
                 *     var plaintext  = CryptoJS.RC4Drop.decrypt(ciphertext, key, cfg);
                 */
                C.RC4Drop = StreamCipher._createHelper(RC4Drop);
            }());


            return CryptoJS.RC4;

        }));

        /***/
    },

    /***/ 243:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(216), __webpack_require__(217), __webpack_require__(227), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./enc-base64", "./md5", "./evpkdf", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var StreamCipher = C_lib.StreamCipher;
                var C_algo = C.algo;

                // Reusable objects
                var S = [];
                var C_ = [];
                var G = [];

                /**
                 * Rabbit stream cipher algorithm
                 */
                var Rabbit = C_algo.Rabbit = StreamCipher.extend({
                    _doReset: function () {
                        // Shortcuts
                        var K = this._key.words;
                        var iv = this.cfg.iv;

                        // Swap endian
                        for (var i = 0; i < 4; i++) {
                            K[i] = (((K[i] << 8) | (K[i] >>> 24)) & 0x00ff00ff) |
                                (((K[i] << 24) | (K[i] >>> 8)) & 0xff00ff00);
                        }

                        // Generate initial state values
                        var X = this._X = [
                            K[0], (K[3] << 16) | (K[2] >>> 16),
                            K[1], (K[0] << 16) | (K[3] >>> 16),
                            K[2], (K[1] << 16) | (K[0] >>> 16),
                            K[3], (K[2] << 16) | (K[1] >>> 16)
                        ];

                        // Generate initial counter values
                        var C = this._C = [
                            (K[2] << 16) | (K[2] >>> 16), (K[0] & 0xffff0000) | (K[1] & 0x0000ffff),
                            (K[3] << 16) | (K[3] >>> 16), (K[1] & 0xffff0000) | (K[2] & 0x0000ffff),
                            (K[0] << 16) | (K[0] >>> 16), (K[2] & 0xffff0000) | (K[3] & 0x0000ffff),
                            (K[1] << 16) | (K[1] >>> 16), (K[3] & 0xffff0000) | (K[0] & 0x0000ffff)
                        ];

                        // Carry bit
                        this._b = 0;

                        // Iterate the system four times
                        for (var i = 0; i < 4; i++) {
                            nextState.call(this);
                        }

                        // Modify the counters
                        for (var i = 0; i < 8; i++) {
                            C[i] ^= X[(i + 4) & 7];
                        }

                        // IV setup
                        if (iv) {
                            // Shortcuts
                            var IV = iv.words;
                            var IV_0 = IV[0];
                            var IV_1 = IV[1];

                            // Generate four subvectors
                            var i0 = (((IV_0 << 8) | (IV_0 >>> 24)) & 0x00ff00ff) | (((IV_0 << 24) | (IV_0 >>> 8)) & 0xff00ff00);
                            var i2 = (((IV_1 << 8) | (IV_1 >>> 24)) & 0x00ff00ff) | (((IV_1 << 24) | (IV_1 >>> 8)) & 0xff00ff00);
                            var i1 = (i0 >>> 16) | (i2 & 0xffff0000);
                            var i3 = (i2 << 16) | (i0 & 0x0000ffff);

                            // Modify counter values
                            C[0] ^= i0;
                            C[1] ^= i1;
                            C[2] ^= i2;
                            C[3] ^= i3;
                            C[4] ^= i0;
                            C[5] ^= i1;
                            C[6] ^= i2;
                            C[7] ^= i3;

                            // Iterate the system four times
                            for (var i = 0; i < 4; i++) {
                                nextState.call(this);
                            }
                        }
                    },

                    _doProcessBlock: function (M, offset) {
                        // Shortcut
                        var X = this._X;

                        // Iterate the system
                        nextState.call(this);

                        // Generate four keystream words
                        S[0] = X[0] ^ (X[5] >>> 16) ^ (X[3] << 16);
                        S[1] = X[2] ^ (X[7] >>> 16) ^ (X[5] << 16);
                        S[2] = X[4] ^ (X[1] >>> 16) ^ (X[7] << 16);
                        S[3] = X[6] ^ (X[3] >>> 16) ^ (X[1] << 16);

                        for (var i = 0; i < 4; i++) {
                            // Swap endian
                            S[i] = (((S[i] << 8) | (S[i] >>> 24)) & 0x00ff00ff) |
                                (((S[i] << 24) | (S[i] >>> 8)) & 0xff00ff00);

                            // Encrypt
                            M[offset + i] ^= S[i];
                        }
                    },

                    blockSize: 128 / 32,

                    ivSize: 64 / 32
                });

                function nextState() {
                    // Shortcuts
                    var X = this._X;
                    var C = this._C;

                    // Save old counter values
                    for (var i = 0; i < 8; i++) {
                        C_[i] = C[i];
                    }

                    // Calculate new counter values
                    C[0] = (C[0] + 0x4d34d34d + this._b) | 0;
                    C[1] = (C[1] + 0xd34d34d3 + ((C[0] >>> 0) < (C_[0] >>> 0) ? 1 : 0)) | 0;
                    C[2] = (C[2] + 0x34d34d34 + ((C[1] >>> 0) < (C_[1] >>> 0) ? 1 : 0)) | 0;
                    C[3] = (C[3] + 0x4d34d34d + ((C[2] >>> 0) < (C_[2] >>> 0) ? 1 : 0)) | 0;
                    C[4] = (C[4] + 0xd34d34d3 + ((C[3] >>> 0) < (C_[3] >>> 0) ? 1 : 0)) | 0;
                    C[5] = (C[5] + 0x34d34d34 + ((C[4] >>> 0) < (C_[4] >>> 0) ? 1 : 0)) | 0;
                    C[6] = (C[6] + 0x4d34d34d + ((C[5] >>> 0) < (C_[5] >>> 0) ? 1 : 0)) | 0;
                    C[7] = (C[7] + 0xd34d34d3 + ((C[6] >>> 0) < (C_[6] >>> 0) ? 1 : 0)) | 0;
                    this._b = (C[7] >>> 0) < (C_[7] >>> 0) ? 1 : 0;

                    // Calculate the g-values
                    for (var i = 0; i < 8; i++) {
                        var gx = X[i] + C[i];

                        // Construct high and low argument for squaring
                        var ga = gx & 0xffff;
                        var gb = gx >>> 16;

                        // Calculate high and low result of squaring
                        var gh = ((((ga * ga) >>> 17) + ga * gb) >>> 15) + gb * gb;
                        var gl = (((gx & 0xffff0000) * gx) | 0) + (((gx & 0x0000ffff) * gx) | 0);

                        // High XOR low
                        G[i] = gh ^ gl;
                    }

                    // Calculate new state values
                    X[0] = (G[0] + ((G[7] << 16) | (G[7] >>> 16)) + ((G[6] << 16) | (G[6] >>> 16))) | 0;
                    X[1] = (G[1] + ((G[0] << 8) | (G[0] >>> 24)) + G[7]) | 0;
                    X[2] = (G[2] + ((G[1] << 16) | (G[1] >>> 16)) + ((G[0] << 16) | (G[0] >>> 16))) | 0;
                    X[3] = (G[3] + ((G[2] << 8) | (G[2] >>> 24)) + G[1]) | 0;
                    X[4] = (G[4] + ((G[3] << 16) | (G[3] >>> 16)) + ((G[2] << 16) | (G[2] >>> 16))) | 0;
                    X[5] = (G[5] + ((G[4] << 8) | (G[4] >>> 24)) + G[3]) | 0;
                    X[6] = (G[6] + ((G[5] << 16) | (G[5] >>> 16)) + ((G[4] << 16) | (G[4] >>> 16))) | 0;
                    X[7] = (G[7] + ((G[6] << 8) | (G[6] >>> 24)) + G[5]) | 0;
                }

                /**
                 * Shortcut functions to the cipher's object interface.
                 *
                 * @example
                 *
                 *     var ciphertext = CryptoJS.Rabbit.encrypt(message, key, cfg);
                 *     var plaintext  = CryptoJS.Rabbit.decrypt(ciphertext, key, cfg);
                 */
                C.Rabbit = StreamCipher._createHelper(Rabbit);
            }());


            return CryptoJS.Rabbit;

        }));

        /***/
    },

    /***/ 244:
    /***/ function (module, exports, __webpack_require__) {

        ;(function (root, factory, undef) {
            if (true) {
                // CommonJS
                module.exports = exports = factory(__webpack_require__(212), __webpack_require__(216), __webpack_require__(217), __webpack_require__(227), __webpack_require__(228));
            }
            else if (typeof define === "function" && define.amd) {
                // AMD
                define(["./core", "./enc-base64", "./md5", "./evpkdf", "./cipher-core"], factory);
            }
            else {
                // Global (browser)
                factory(root.CryptoJS);
            }
        }(this, function (CryptoJS) {

            (function () {
                // Shortcuts
                var C = CryptoJS;
                var C_lib = C.lib;
                var StreamCipher = C_lib.StreamCipher;
                var C_algo = C.algo;

                // Reusable objects
                var S = [];
                var C_ = [];
                var G = [];

                /**
                 * Rabbit stream cipher algorithm.
                 *
                 * This is a legacy version that neglected to convert the key to little-endian.
                 * This error doesn't affect the cipher's security,
                 * but it does affect its compatibility with other implementations.
                 */
                var RabbitLegacy = C_algo.RabbitLegacy = StreamCipher.extend({
                    _doReset: function () {
                        // Shortcuts
                        var K = this._key.words;
                        var iv = this.cfg.iv;

                        // Generate initial state values
                        var X = this._X = [
                            K[0], (K[3] << 16) | (K[2] >>> 16),
                            K[1], (K[0] << 16) | (K[3] >>> 16),
                            K[2], (K[1] << 16) | (K[0] >>> 16),
                            K[3], (K[2] << 16) | (K[1] >>> 16)
                        ];

                        // Generate initial counter values
                        var C = this._C = [
                            (K[2] << 16) | (K[2] >>> 16), (K[0] & 0xffff0000) | (K[1] & 0x0000ffff),
                            (K[3] << 16) | (K[3] >>> 16), (K[1] & 0xffff0000) | (K[2] & 0x0000ffff),
                            (K[0] << 16) | (K[0] >>> 16), (K[2] & 0xffff0000) | (K[3] & 0x0000ffff),
                            (K[1] << 16) | (K[1] >>> 16), (K[3] & 0xffff0000) | (K[0] & 0x0000ffff)
                        ];

                        // Carry bit
                        this._b = 0;

                        // Iterate the system four times
                        for (var i = 0; i < 4; i++) {
                            nextState.call(this);
                        }

                        // Modify the counters
                        for (var i = 0; i < 8; i++) {
                            C[i] ^= X[(i + 4) & 7];
                        }

                        // IV setup
                        if (iv) {
                            // Shortcuts
                            var IV = iv.words;
                            var IV_0 = IV[0];
                            var IV_1 = IV[1];

                            // Generate four subvectors
                            var i0 = (((IV_0 << 8) | (IV_0 >>> 24)) & 0x00ff00ff) | (((IV_0 << 24) | (IV_0 >>> 8)) & 0xff00ff00);
                            var i2 = (((IV_1 << 8) | (IV_1 >>> 24)) & 0x00ff00ff) | (((IV_1 << 24) | (IV_1 >>> 8)) & 0xff00ff00);
                            var i1 = (i0 >>> 16) | (i2 & 0xffff0000);
                            var i3 = (i2 << 16) | (i0 & 0x0000ffff);

                            // Modify counter values
                            C[0] ^= i0;
                            C[1] ^= i1;
                            C[2] ^= i2;
                            C[3] ^= i3;
                            C[4] ^= i0;
                            C[5] ^= i1;
                            C[6] ^= i2;
                            C[7] ^= i3;

                            // Iterate the system four times
                            for (var i = 0; i < 4; i++) {
                                nextState.call(this);
                            }
                        }
                    },

                    _doProcessBlock: function (M, offset) {
                        // Shortcut
                        var X = this._X;

                        // Iterate the system
                        nextState.call(this);

                        // Generate four keystream words
                        S[0] = X[0] ^ (X[5] >>> 16) ^ (X[3] << 16);
                        S[1] = X[2] ^ (X[7] >>> 16) ^ (X[5] << 16);
                        S[2] = X[4] ^ (X[1] >>> 16) ^ (X[7] << 16);
                        S[3] = X[6] ^ (X[3] >>> 16) ^ (X[1] << 16);

                        for (var i = 0; i < 4; i++) {
                            // Swap endian
                            S[i] = (((S[i] << 8) | (S[i] >>> 24)) & 0x00ff00ff) |
                                (((S[i] << 24) | (S[i] >>> 8)) & 0xff00ff00);

                            // Encrypt
                            M[offset + i] ^= S[i];
                        }
                    },

                    blockSize: 128 / 32,

                    ivSize: 64 / 32
                });

                function nextState() {
                    // Shortcuts
                    var X = this._X;
                    var C = this._C;

                    // Save old counter values
                    for (var i = 0; i < 8; i++) {
                        C_[i] = C[i];
                    }

                    // Calculate new counter values
                    C[0] = (C[0] + 0x4d34d34d + this._b) | 0;
                    C[1] = (C[1] + 0xd34d34d3 + ((C[0] >>> 0) < (C_[0] >>> 0) ? 1 : 0)) | 0;
                    C[2] = (C[2] + 0x34d34d34 + ((C[1] >>> 0) < (C_[1] >>> 0) ? 1 : 0)) | 0;
                    C[3] = (C[3] + 0x4d34d34d + ((C[2] >>> 0) < (C_[2] >>> 0) ? 1 : 0)) | 0;
                    C[4] = (C[4] + 0xd34d34d3 + ((C[3] >>> 0) < (C_[3] >>> 0) ? 1 : 0)) | 0;
                    C[5] = (C[5] + 0x34d34d34 + ((C[4] >>> 0) < (C_[4] >>> 0) ? 1 : 0)) | 0;
                    C[6] = (C[6] + 0x4d34d34d + ((C[5] >>> 0) < (C_[5] >>> 0) ? 1 : 0)) | 0;
                    C[7] = (C[7] + 0xd34d34d3 + ((C[6] >>> 0) < (C_[6] >>> 0) ? 1 : 0)) | 0;
                    this._b = (C[7] >>> 0) < (C_[7] >>> 0) ? 1 : 0;

                    // Calculate the g-values
                    for (var i = 0; i < 8; i++) {
                        var gx = X[i] + C[i];

                        // Construct high and low argument for squaring
                        var ga = gx & 0xffff;
                        var gb = gx >>> 16;

                        // Calculate high and low result of squaring
                        var gh = ((((ga * ga) >>> 17) + ga * gb) >>> 15) + gb * gb;
                        var gl = (((gx & 0xffff0000) * gx) | 0) + (((gx & 0x0000ffff) * gx) | 0);

                        // High XOR low
                        G[i] = gh ^ gl;
                    }

                    // Calculate new state values
                    X[0] = (G[0] + ((G[7] << 16) | (G[7] >>> 16)) + ((G[6] << 16) | (G[6] >>> 16))) | 0;
                    X[1] = (G[1] + ((G[0] << 8) | (G[0] >>> 24)) + G[7]) | 0;
                    X[2] = (G[2] + ((G[1] << 16) | (G[1] >>> 16)) + ((G[0] << 16) | (G[0] >>> 16))) | 0;
                    X[3] = (G[3] + ((G[2] << 8) | (G[2] >>> 24)) + G[1]) | 0;
                    X[4] = (G[4] + ((G[3] << 16) | (G[3] >>> 16)) + ((G[2] << 16) | (G[2] >>> 16))) | 0;
                    X[5] = (G[5] + ((G[4] << 8) | (G[4] >>> 24)) + G[3]) | 0;
                    X[6] = (G[6] + ((G[5] << 16) | (G[5] >>> 16)) + ((G[4] << 16) | (G[4] >>> 16))) | 0;
                    X[7] = (G[7] + ((G[6] << 8) | (G[6] >>> 24)) + G[5]) | 0;
                }

                /**
                 * Shortcut functions to the cipher's object interface.
                 *
                 * @example
                 *
                 *     var ciphertext = CryptoJS.RabbitLegacy.encrypt(message, key, cfg);
                 *     var plaintext  = CryptoJS.RabbitLegacy.decrypt(ciphertext, key, cfg);
                 */
                C.RabbitLegacy = StreamCipher._createHelper(RabbitLegacy);
            }());


            return CryptoJS.RabbitLegacy;

        }));

        /***/
    },

    /***/ 247:
    /***/ function (module, exports, __webpack_require__) {

        'use strict';

        module.exports = __webpack_require__(248);

        /***/
    },

    /***/ 248:
    /***/ function (module, exports, __webpack_require__) {

        'use strict';

        var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) {
            return typeof obj;
        } : function (obj) {
            return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj;
        };

        var _version = '1.4.2';
        var _code = __webpack_require__(207).code;
        var _utils = __webpack_require__(206).utils;
        var _msg = __webpack_require__(249);
        var _message = _msg._msg;
        var _msgHash = {};
        var Queue = __webpack_require__(250).Queue;
        var CryptoJS = __webpack_require__(211);
        var _ = __webpack_require__(183);
        var stropheConn = null;

        window.URL = window.URL || window.webkitURL || window.mozURL || window.msURL;

        if (window.XDomainRequest) {
            // not support ie8 send is not a function , canot
            // case send is object, doesn't has a attr of call
            // XDomainRequest.prototype.oldsend = XDomainRequest.prototype.send;
            // XDomainRequest.prototype.send = function () {
            //     XDomainRequest.prototype.oldsend.call(this, arguments);
            //     this.readyState = 2;
            // };
        }

        Strophe.Connection.prototype._sasl_auth1_cb = function (elem) {
            // save stream:features for future usage
            this.features = elem;
            var i, child;
            for (i = 0; i < elem.childNodes.length; i++) {
                child = elem.childNodes[i];
                if (child.nodeName == 'bind') {
                    this.do_bind = true;
                }

                if (child.nodeName == 'session') {
                    this.do_session = true;
                }
            }

            if (!this.do_bind) {
                this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
                return false;
            } else {
                this._addSysHandler(this._sasl_bind_cb.bind(this), null, null, null, "_bind_auth_2");

                var resource = Strophe.getResourceFromJid(this.jid);
                if (resource) {
                    // this.send($iq({type: "set", id: "_bind_auth_2"})
                    //     .c('bind', {xmlns: Strophe.NS.BIND})
                    //     .c('resource', {}).t(resource).tree());
                    var device_uuid = "device_uuid";
                    if (this.options.isMultiLoginSessions) {
                        device_uuid = new Date().getTime() + Math.floor(Math.random().toFixed(6) * 1000000);
                    }
                    try {
                        this.send($iq({
                            type: "set",
                            id: "_bind_auth_2"
                        }).c('bind', {xmlns: Strophe.NS.BIND}).c('resource', {}).t(resource).up().c('os').t('webim').up().c('device_uuid').t(device_uuid).up().c('is_manual_login').t('true').tree());
                    } catch (e) {
                        console.log("Bind Error: ", e.message);
                    }
                } else {
                    this.send($iq({type: "set", id: "_bind_auth_2"}).c('bind', {xmlns: Strophe.NS.BIND}).tree());
                }
            }
            return false;
        };

        Strophe.Request.prototype._newXHR = function () {
            var xhr = _utils.xmlrequest(true);
            if (xhr.overrideMimeType) {
                xhr.overrideMimeType('text/xml');
            }
            // use Function.bind() to prepend ourselves as an argument
            xhr.onreadystatechange = this.func.bind(null, this);
            return xhr;
        };

        Strophe.Websocket.prototype._closeSocket = function () {
            if (this.socket) {
                var me = this;
                setTimeout(function () {
                    try {
                        me.socket.close();
                    } catch (e) {
                    }
                }, 0);
            } else {
                this.socket = null;
            }
        };

        /**
         *
         * Strophe.Websocket has a bug while logout:
         * 1.send: <presence xmlns='jabber:client' type='unavailable'/> is ok;
         * 2.send: <close xmlns='urn:ietf:params:xml:ns:xmpp-framing'/> will cause a problem,log as follows:
         * WebSocket connection to 'ws://im-api.easemob.com/ws/' failed: Data frame received after close_connect @ strophe.js:5292connect @ strophe.js:2491_login @ websdk-1.1.2.js:278suc @ websdk-1.1.2.js:636xhr.onreadystatechange @ websdk-1.1.2.js:2582
         * 3 "Websocket error [object Event]"
         * _changeConnectStatus
         * onError Object {type: 7, msg: "The WebSocket connection could not be established or was disconnected.", reconnect: true}
         *
         * this will trigger socket.onError, therefore _doDisconnect again.
         * Fix it by overide  _onMessage
         */
        Strophe.Websocket.prototype._onMessage = function (message) {
            logMessage(message);
            // 获取Resource
            var data = message.data;
            if (data.indexOf('<jid>') > 0) {
                var start = data.indexOf('<jid>'),
                    end = data.indexOf('</jid>'),
                    data = data.substring(start + 5, end);
                stropheConn.setJid(data);
            }

            var elem, data;
            // check for closing stream
            // var close = '<close xmlns="urn:ietf:params:xml:ns:xmpp-framing" />';
            // if (message.data === close) {
            //     this._conn.rawInput(close);
            //     this._conn.xmlInput(message);
            //     if (!this._conn.disconnecting) {
            //         this._conn._doDisconnect();
            //     }
            //     return;
            //
            // send and receive close xml: <close xmlns='urn:ietf:params:xml:ns:xmpp-framing'/>
            // so we can't judge whether message.data equals close by === simply.
            if (message.data.indexOf("<close ") === 0) {
                elem = new DOMParser().parseFromString(message.data, "text/xml").documentElement;
                var see_uri = elem.getAttribute("see-other-uri");
                if (see_uri) {
                    this._conn._changeConnectStatus(Strophe.Status.REDIRECT, "Received see-other-uri, resetting connection");
                    this._conn.reset();
                    this._conn.service = see_uri;
                    this._connect();
                } else {
                    // if (!this._conn.disconnecting) {
                    this._conn._doDisconnect("receive <close> from server");
                    // }
                }
                return;
            } else if (message.data.search("<open ") === 0) {
                // This handles stream restarts
                elem = new DOMParser().parseFromString(message.data, "text/xml").documentElement;
                if (!this._handleStreamStart(elem)) {
                    return;
                }
            } else {
                data = this._streamWrap(message.data);
                elem = new DOMParser().parseFromString(data, "text/xml").documentElement;
            }

            if (this._check_streamerror(elem, Strophe.Status.ERROR)) {
                return;
            }

            //handle unavailable presence stanza before disconnecting
            if (this._conn.disconnecting && elem.firstChild.nodeName === "presence" && elem.firstChild.getAttribute("type") === "unavailable") {
                this._conn.xmlInput(elem);
                this._conn.rawInput(Strophe.serialize(elem));
                // if we are already disconnecting we will ignore the unavailable stanza and
                // wait for the </stream:stream> tag before we close the connection
                return;
            }
            this._conn._dataRecv(elem, message.data);
        };

        var _listenNetwork = function _listenNetwork(onlineCallback, offlineCallback) {

            if (window.addEventListener) {
                window.addEventListener('online', onlineCallback);
                window.addEventListener('offline', offlineCallback);
            } else if (window.attachEvent) {
                if (document.body) {
                    document.body.attachEvent('ononline', onlineCallback);
                    document.body.attachEvent('onoffline', offlineCallback);
                } else {
                    window.attachEvent('load', function () {
                        document.body.attachEvent('ononline', onlineCallback);
                        document.body.attachEvent('onoffline', offlineCallback);
                    });
                }
            } else {
                /*var onlineTmp = window.ononline;
	         var offlineTmp = window.onoffline;
	          window.attachEvent('ononline', function () {
	         try {
	         typeof onlineTmp === 'function' && onlineTmp();
	         } catch ( e ) {}
	         onlineCallback();
	         });
	         window.attachEvent('onoffline', function () {
	         try {
	         typeof offlineTmp === 'function' && offlineTmp();
	         } catch ( e ) {}
	         offlineCallback();
	         });*/
            }
        };

        var _parseRoom = function _parseRoom(result) {
            var rooms = [];
            var items = result.getElementsByTagName('item');
            if (items) {
                for (var i = 0; i < items.length; i++) {
                    var item = items[i];
                    var roomJid = item.getAttribute('jid');
                    var tmp = roomJid.split('@')[0];
                    var room = {
                        jid: roomJid,
                        name: item.getAttribute('name'),
                        roomId: tmp.split('_')[1]
                    };
                    rooms.push(room);
                }
            }
            return rooms;
        };

        var _parseRoomOccupants = function _parseRoomOccupants(result) {
            var occupants = [];
            var items = result.getElementsByTagName('item');
            if (items) {
                for (var i = 0; i < items.length; i++) {
                    var item = items[i];
                    var room = {
                        jid: item.getAttribute('jid'),
                        name: item.getAttribute('name')
                    };
                    occupants.push(room);
                }
            }
            return occupants;
        };

        var _parseResponseMessage = function _parseResponseMessage(msginfo) {
            var parseMsgData = {errorMsg: true, data: []};

            var msgBodies = msginfo.getElementsByTagName('body');
            if (msgBodies) {
                for (var i = 0; i < msgBodies.length; i++) {
                    var msgBody = msgBodies[i];
                    var childNodes = msgBody.childNodes;
                    if (childNodes && childNodes.length > 0) {
                        var childNode = msgBody.childNodes[0];
                        if (childNode.nodeType == Strophe.ElementType.TEXT) {
                            var jsondata = childNode.wholeText || childNode.nodeValue;
                            jsondata = jsondata.replace('\n', '<br>');
                            try {
                                var data = eval('(' + jsondata + ')');
                                parseMsgData.errorMsg = false;
                                parseMsgData.data = [data];
                            } catch (e) {
                            }
                        }
                    }
                }

                var delayTags = msginfo.getElementsByTagName('delay');
                if (delayTags && delayTags.length > 0) {
                    var delayTag = delayTags[0];
                    var delayMsgTime = delayTag.getAttribute('stamp');
                    if (delayMsgTime) {
                        parseMsgData.delayTimeStamp = delayMsgTime;
                    }
                }
            } else {
                var childrens = msginfo.childNodes;
                if (childrens && childrens.length > 0) {
                    var child = msginfo.childNodes[0];
                    if (child.nodeType == Strophe.ElementType.TEXT) {
                        try {
                            var data = eval('(' + child.nodeValue + ')');
                            parseMsgData.errorMsg = false;
                            parseMsgData.data = [data];
                        } catch (e) {
                        }
                    }
                }
            }
            return parseMsgData;
        };

        var _parseNameFromJidFn = function _parseNameFromJidFn(jid, domain) {
            domain = domain || '';
            var tempstr = jid;
            var findex = tempstr.indexOf('_');

            if (findex !== -1) {
                tempstr = tempstr.substring(findex + 1);
            }
            var atindex = tempstr.indexOf('@' + domain);
            if (atindex !== -1) {
                tempstr = tempstr.substring(0, atindex);
            }
            return tempstr;
        };

        var _parseFriend = function _parseFriend(queryTag, conn, from) {
            var rouster = [];
            var items = queryTag.getElementsByTagName('item');
            if (items) {
                for (var i = 0; i < items.length; i++) {
                    var item = items[i];
                    var jid = item.getAttribute('jid');
                    if (!jid) {
                        continue;
                    }
                    var subscription = item.getAttribute('subscription');
                    var friend = {
                        subscription: subscription,
                        jid: jid
                    };
                    var ask = item.getAttribute('ask');
                    if (ask) {
                        friend.ask = ask;
                    }
                    var name = item.getAttribute('name');
                    if (name) {
                        friend.name = name;
                    } else {
                        var n = _parseNameFromJidFn(jid);
                        friend.name = n;
                    }
                    var groups = [];
                    Strophe.forEachChild(item, 'group', function (group) {
                        groups.push(Strophe.getText(group));
                    });
                    friend.groups = groups;
                    rouster.push(friend);
                    // B同意之后 -> B订阅A
                    // fix: 含有ask标示的好友代表已经发送过反向订阅消息，不需要再次发送。
                    if (conn && subscription == 'from' && !ask) {
                        conn.subscribe({
                            toJid: jid,
                            message: "[resp:true]"
                        });
                    }

                    if (conn && subscription == 'to') {
                        conn.subscribed({
                            toJid: jid
                        });
                    }
                }
            }
            return rouster;
        };

        var _login = function _login(options, conn) {
            var accessToken = options.access_token || '';
            if (accessToken == '') {
                var loginfo = _utils.stringify(options);
                conn.onError({
                    type: _code.WEBIM_CONNCTION_OPEN_USERGRID_ERROR,
                    data: options
                });
                return;
            }
            conn.context.accessToken = options.access_token;
            conn.context.accessTokenExpires = options.expires_in;
            if (conn.isOpening() && conn.context.stropheConn) {
                stropheConn = conn.getStrophe();
            } else if (conn.isOpened() && conn.context.stropheConn) {
                // return;
                stropheConn = conn.getStrophe();
            } else {
                stropheConn = conn.getStrophe();
            }
            var callback = function callback(status, msg) {
                _loginCallback(status, msg, conn);
            };

            conn.context.stropheConn = stropheConn;
            if (conn.route) {
                stropheConn.connect(conn.context.jid, '$t$' + accessToken, callback, conn.wait, conn.hold, conn.route);
            } else {
                stropheConn.connect(conn.context.jid, '$t$' + accessToken, callback, conn.wait, conn.hold);
            }
        };

        var _parseMessageType = function _parseMessageType(msginfo) {
            var receiveinfo = msginfo.getElementsByTagName('received'),
                inviteinfo = msginfo.getElementsByTagName('invite'),
                deliveryinfo = msginfo.getElementsByTagName('delivery'),
                acked = msginfo.getElementsByTagName('acked'),
                error = msginfo.getElementsByTagName('error'),
                msgtype = 'normal';
            if (receiveinfo && receiveinfo.length > 0 && receiveinfo[0].namespaceURI === 'urn:xmpp:receipts') {

                msgtype = 'received';
            } else if (inviteinfo && inviteinfo.length > 0) {

                msgtype = 'invite';
            } else if (deliveryinfo && deliveryinfo.length > 0) {

                msgtype = 'delivery'; // 消息送达
            } else if (acked && acked.length) {

                msgtype = 'acked'; // 消息已读
            } else if (error && error.length) {

                var errorItem = error[0],
                    userMuted = errorItem.getElementsByTagName('user-muted');

                if (userMuted && userMuted.length) {

                    msgtype = 'userMuted';
                }
            }
            return msgtype;
        };

        var _handleMessageQueue = function _handleMessageQueue(conn) {
            for (var i in _msgHash) {
                if (_msgHash.hasOwnProperty(i)) {
                    _msgHash[i].send(conn);
                }
            }
        };

        var _loginCallback = function _loginCallback(status, msg, conn) {
            var conflict, error;

            if (msg === 'conflict') {
                conflict = true;
                conn.close();
            }

            if (status == Strophe.Status.CONNFAIL) {
                //client offline, ping/pong timeout, server quit, server offline
                error = {
                    type: _code.WEBIM_CONNCTION_SERVER_CLOSE_ERROR,
                    msg: msg,
                    reconnect: true
                };

                conflict && (error.conflict = true);
                conn.onError(error);
            } else if (status == Strophe.Status.ATTACHED || status == Strophe.Status.CONNECTED) {
                // client should limit the speed of sending ack messages  up to 5/s
                conn.autoReconnectNumTotal = 0;
                conn.intervalId = setInterval(function () {
                    conn.handelSendQueue();
                }, 200);
                var handleMessage = function handleMessage(msginfo) {
                    var delivery = msginfo.getElementsByTagName('delivery');
                    var acked = msginfo.getElementsByTagName('acked');
                    if (delivery.length) {
                        conn.handleDeliveredMessage(msginfo);
                        return true;
                    }
                    if (acked.length) {
                        conn.handleAckedMessage(msginfo);
                        return true;
                    }
                    var type = _parseMessageType(msginfo);
                    switch (type) {
                        case "received":
                            conn.handleReceivedMessage(msginfo);
                            return true;
                        case "invite":
                            conn.handleInviteMessage(msginfo);
                            return true;
                        case "delivery":
                            conn.handleDeliveredMessage(msginfo);
                            return true;
                        case "acked":
                            conn.handleAckedMessage(msginfo);
                            return true;
                        case "userMuted":
                            conn.handleMutedMessage(msginfo);
                            return true;
                        default:
                            conn.handleMessage(msginfo);
                            return true;
                    }
                };
                var handlePresence = function handlePresence(msginfo) {
                    conn.handlePresence(msginfo);
                    return true;
                };
                var handlePing = function handlePing(msginfo) {
                    conn.handlePing(msginfo);
                    return true;
                };
                var handleIqRoster = function handleIqRoster(msginfo) {
                    conn.handleIqRoster(msginfo);
                    return true;
                };
                var handleIqPrivacy = function handleIqPrivacy(msginfo) {
                    conn.handleIqPrivacy(msginfo);
                    return true;
                };
                var handleIq = function handleIq(msginfo) {
                    conn.handleIq(msginfo);
                    return true;
                };

                conn.addHandler(handleMessage, null, 'message', null, null, null);
                conn.addHandler(handlePresence, null, 'presence', null, null, null);
                conn.addHandler(handlePing, 'urn:xmpp:ping', 'iq', 'get', null, null);
                conn.addHandler(handleIqRoster, 'jabber:iq:roster', 'iq', 'set', null, null);
                conn.addHandler(handleIqPrivacy, 'jabber:iq:privacy', 'iq', 'set', null, null);
                conn.addHandler(handleIq, null, 'iq', null, null, null);

                conn.registerConfrIQHandler && conn.registerConfrIQHandler();

                conn.context.status = _code.STATUS_OPENED;

                var supportRecMessage = [_code.WEBIM_MESSAGE_REC_TEXT, _code.WEBIM_MESSAGE_REC_EMOJI];

                if (_utils.isCanDownLoadFile) {
                    supportRecMessage.push(_code.WEBIM_MESSAGE_REC_PHOTO);
                    supportRecMessage.push(_code.WEBIM_MESSAGE_REC_AUDIO_FILE);
                }
                var supportSedMessage = [_code.WEBIM_MESSAGE_SED_TEXT];
                if (_utils.isCanUploadFile) {
                    supportSedMessage.push(_code.WEBIM_MESSAGE_REC_PHOTO);
                    supportSedMessage.push(_code.WEBIM_MESSAGE_REC_AUDIO_FILE);
                }
                conn.notifyVersion();
                conn.retry && _handleMessageQueue(conn);
                conn.heartBeat();
                conn.isAutoLogin && conn.setPresence();

                try {
                    if (conn.unSendMsgArr.length > 0) {
                        for (var i in conn.unSendMsgArr) {
                            var dom = conn.unSendMsgArr[i];
                            conn.sendCommand(dom);
                            delete conn.unSendMsgArr[i];
                        }
                    }
                } catch (e) {
                    console.error(e.message);
                }
                conn.offLineSendConnecting = false;
                conn.logOut = false;

                conn.onOpened({
                    canReceive: supportRecMessage,
                    canSend: supportSedMessage,
                    accessToken: conn.context.accessToken
                });
            } else if (status == Strophe.Status.DISCONNECTING) {
                if (conn.isOpened()) {
                    conn.stopHeartBeat();
                    conn.context.status = _code.STATUS_CLOSING;

                    error = {
                        type: _code.WEBIM_CONNCTION_SERVER_CLOSE_ERROR,
                        msg: msg,
                        reconnect: true
                    };

                    conflict && (error.conflict = true);
                    conn.onError(error);
                }
            } else if (status == Strophe.Status.DISCONNECTED) {
                if (conn.isOpened()) {
                    if (conn.autoReconnectNumTotal < conn.autoReconnectNumMax) {
                        conn.reconnect();
                        return;
                    } else {
                        error = {
                            type: _code.WEBIM_CONNCTION_DISCONNECTED
                        };
                        conn.onError(error);
                    }
                }
                conn.context.status = _code.STATUS_CLOSED;
                conn.clear();
                conn.onClosed();
            } else if (status == Strophe.Status.AUTHFAIL) {
                error = {
                    type: _code.WEBIM_CONNCTION_AUTH_ERROR
                };

                conflict && (error.conflict = true);
                conn.onError(error);
                conn.clear();
            } else if (status == Strophe.Status.ERROR) {
                conn.context.status = _code.STATUS_ERROR;
                error = {
                    type: _code.WEBIM_CONNCTION_SERVER_ERROR
                };

                conflict && (error.conflict = true);
                conn.onError(error);
            }
            conn.context.status_now = status;
        };

        var _getJid = function _getJid(options, conn) {
            var jid = options.toJid || '';

            if (jid === '') {
                var appKey = conn.context.appKey || '';
                var toJid = appKey + '_' + options.to + '@' + conn.domain;

                if (options.resource) {
                    toJid = toJid + '/' + options.resource;
                }
                jid = toJid;
            }
            return jid;
        };

        var _getJidByName = function _getJidByName(name, conn) {
            var options = {
                to: name
            };
            return _getJid(options, conn);
        };

        var _validCheck = function _validCheck(options, conn) {
            options = options || {};

            if (options.user == '') {
                conn.onError({
                    type: _code.WEBIM_CONNCTION_USER_NOT_ASSIGN_ERROR
                });
                return false;
            }

            var user = options.user + '' || '';
            var appKey = options.appKey || '';
            var devInfos = appKey.split('#');

            if (devInfos.length !== 2) {
                conn.onError({
                    type: _code.WEBIM_CONNCTION_APPKEY_NOT_ASSIGN_ERROR
                });
                return false;
            }
            var orgName = devInfos[0];
            var appName = devInfos[1];

            if (!orgName) {
                conn.onError({
                    type: _code.WEBIM_CONNCTION_APPKEY_NOT_ASSIGN_ERROR
                });
                return false;
            }
            if (!appName) {
                conn.onError({
                    type: _code.WEBIM_CONNCTION_APPKEY_NOT_ASSIGN_ERROR
                });
                return false;
            }

            var jid = appKey + '_' + user.toLowerCase() + '@' + conn.domain,
                resource = options.resource || 'webim';

            conn.context.jid = jid + '/' + resource;
            conn.context.userId = user;
            conn.context.appKey = appKey;
            conn.context.appName = appName;
            conn.context.orgName = orgName;

            return true;
        };

        var _getXmppUrl = function _getXmppUrl(baseUrl, https) {
            if (/^(ws|http)s?:\/\/?/.test(baseUrl)) {
                return baseUrl;
            }

            var url = {
                prefix: 'http',
                base: '://' + baseUrl,
                suffix: '/http-bind/'
            };

            if (https && _utils.isSupportWss) {
                url.prefix = 'wss';
                url.suffix = '/ws/';
            } else {
                if (https) {
                    url.prefix = 'https';
                } else if (window.WebSocket) {
                    url.prefix = 'ws';
                    url.suffix = '/ws/';
                }
            }

            return url.prefix + url.base + url.suffix;
        };

        function _deepClone(data) {
            var t = typeof data === 'undefined' ? 'undefined' : _typeof(data),
                o,
                i,
                ni;

            if (t === 'array') {
                o = [];
            } else if (t === 'object') {
                o = {};
            } else {
                return data;
            }

            if (t === 'array') {
                for (i = 0, ni = data.length; i < ni; i++) {
                    o.push(_deepClone(data[i]));
                }
                return o;
            } else if (t === 'object') {
                for (i in data) {
                    o[i] = _deepClone(data[i]);
                }
                return o;
            }
        }

        /**
         * The connection class.
         * @constructor
         * @param {Object} options - 创建连接的初始化参数
         * @param {String} options.url - xmpp服务器的URL
         * @param {String} options.apiUrl - API服务器的URL
         * @param {Boolean} options.isHttpDNS - 防止域名劫持
         * @param {Boolean} options.isMultiLoginSessions - 为true时同一账户可以同时在多个Web页面登录（多标签登录，默认不开启，如有需要请联系商务），为false时同一账号只能在一个Web页面登录
         * @param {Boolean} options.https - 是否启用wss.
         * @param {Number} options.heartBeatWait - 发送心跳包的时间间隔（毫秒）
         * @param {Boolean} options.isAutoLogin - 登录成功后是否自动出席
         * @param {Number} options.autoReconnectNumMax - 掉线后重连的最大次数
         * @param {Number} options.autoReconnectInterval -  掉线后重连的间隔时间（毫秒）
         * @param {Boolean} options.isWindowSDK - 是否运行在WindowsSDK上
         * @param {Boolean} options.encrypt - 是否加密文本消息
         * @param {Boolean} options.delivery - 是否发送delivered ack
         * @returns {Class}  连接实例
         */

        var connection = function connection(options) {
            if (!this instanceof connection) {
                return new connection(options);
            }

            var options = options || {};

            this.isHttpDNS = options.isHttpDNS || false;
            this.isMultiLoginSessions = options.isMultiLoginSessions || false;
            this.wait = options.wait || 30;
            this.hold = options.hold || 1;
            this.retry = options.retry || false;
            this.https = options.https || location.protocol === 'https:';
            this.url = _getXmppUrl(options.url, this.https);
            this.route = options.route || null;
            this.domain = options.domain || 'easemob.com';
            this.inactivity = options.inactivity || 30;
            this.heartBeatWait = options.heartBeatWait || 4500;
            this.maxRetries = options.maxRetries || 5;
            this.isAutoLogin = options.isAutoLogin === false ? false : true;
            this.pollingTime = options.pollingTime || 800;
            this.stropheConn = false;
            this.autoReconnectNumMax = options.autoReconnectNumMax || 0;
            this.autoReconnectNumTotal = 0;
            this.autoReconnectInterval = options.autoReconnectInterval || 0;
            this.context = {status: _code.STATUS_INIT};
            this.sendQueue = new Queue(); //instead of sending message immediately,cache them in this queue
            this.intervalId = null; //clearInterval return value
            this.apiUrl = options.apiUrl || '';
            this.isWindowSDK = options.isWindowSDK || false;
            this.encrypt = options.encrypt || {encrypt: {type: 'none'}};
            this.delivery = options.delivery || false;
            this.saveLocal = options.saveLocal || false;
            this.user = '';
            this.orgName = '';
            this.appName = '';
            this.token = '';
            this.unSendMsgArr = [];
            this.offLineSendConnecting = false;
            this.logOut = false;

            this.dnsArr = ['https://rs.easemob.com', 'https://rsbak.easemob.com', 'http://182.92.174.78', 'http://112.126.66.111']; //http dns server hosts
            this.dnsIndex = 0; //the dns ip used in dnsArr currently
            this.dnsTotal = this.dnsArr.length; //max number of getting dns retries
            this.restHosts = null; //rest server ips
            this.restIndex = 0; //the rest ip used in restHosts currently
            this.restTotal = 0; //max number of getting rest token retries
            this.xmppHosts = null; //xmpp server ips
            this.xmppIndex = 0; //the xmpp ip used in xmppHosts currently
            this.xmppTotal = 0; //max number of creating xmpp server connection(ws/bosh) retries

            this.groupOption = {};

            /*
	     Demo.chatRecord = {
	     targetId: {
	     messages: [
	     {
	     msg: 'msg',
	     type: 'type'
	     },
	     {
	     msg: 'msg',
	     type: 'type'
	     }],
	     brief: 'brief'
	     }
	     }
	     */
        };

        connection.prototype.testInit = function (options) {
            this.orgName = options.orgName;
            this.appName = options.appName;
            this.user = options.user;
            this.token = options.token;
        };

        /**
         * 注册新用户
         * @param {Object} options - 用户信息
         * @param {String} options.username - 用户名
         * @param {String} options.password - 密码
         * @param {String} options.nickname - 用户昵称
         * @param {Function} options.success - 注册成功回调
         * @param {Function} options.error - 注册失败
         */
        connection.prototype.registerUser = function (options) {
            if (location.protocol != 'https:' && this.isHttpDNS) {
                this.dnsIndex = 0;
                this.getHttpDNS(options, 'signup');
            } else {
                this.signup(options);
            }
        };

        /**
         * 处理发送队列
         * @private
         */
        connection.prototype.handelSendQueue = function () {
            var options = this.sendQueue.pop();
            if (options !== null) {
                this.sendReceiptsMessage(options);
            }
        };

        /**
         * 注册监听函数
         * @param {Object} options - 回调函数集合
         * @param {connection~onOpened} options.onOpened - 处理登录的回调
         * @param {connection~onTextMessage} options.onTextMessage - 处理文本消息的回调
         * @param {connection~onEmojiMessage} options.onEmojiMessage - 处理表情消息的回调
         * @param {connection~onPictureMessage} options.onPictureMessage - 处理图片消息的回调
         * @param {connection~onAudioMessage} options.onAudioMessage - 处理音频消息的回调
         * @param {connection~onVideoMessage} options.onVideoMessage - 处理视频消息的回调
         * @param {connection~onFileMessage} options.onFileMessage - 处理文件消息的回调
         * @param {connection~onLocationMessage} options.onLocationMessage - 处理位置消息的回调
         * @param {connection~onCmdMessage} options.onCmdMessage - 处理命令消息的回调
         * @param {connection~onPresence} options.onPresence - 处理Presence消息的回调
         * @param {connection~onError} options.onError - 处理错误消息的回调
         * @param {connection~onReceivedMessage} options.onReceivedMessage - 处理Received消息的回调
         * @param {connection~onInviteMessage} options.onInviteMessage - 处理邀请消息的回调
         * @param {connection~onDeliverdMessage} options.onDeliverdMessage - 处理Delivered ACK消息的回调
         * @param {connection~onReadMessage} options.onReadMessage - 处理Read ACK消息的回调
         * @param {connection~onMutedMessage} options.onMutedMessage - 处理禁言消息的回调
         * @param {connection~onOffline} options.onOffline - 处理断网的回调
         * @param {connection~onOnline} options.onOnline - 处理联网的回调
         * @param {connection~onCreateGroup} options.onCreateGroup - 处理创建群组的回调
         */
        connection.prototype.listen = function (options) {
            /**
             * 登录成功后调用
             * @callback connection~onOpened
             */
            /**
             * 收到文本消息
             * @callback connection~onTextMessage
             */
            /**
             * 收到表情消息
             * @callback connection~onEmojiMessage
             */
            /**
             * 收到图片消息
             * @callback connection~onPictureMessage
             */
            /**
             * 收到音频消息
             * @callback connection~onAudioMessage
             */
            /**
             * 收到视频消息
             * @callback connection~onVideoMessage
             */
            /**
             * 收到文件消息
             * @callback connection~onFileMessage
             */
            /**
             * 收到位置消息
             * @callback connection~onLocationMessage
             */
            /**
             * 收到命令消息
             * @callback connection~onCmdMessage
             */
            /**
             * 收到错误消息
             * @callback connection~onError
             */
            /**
             * 收到Presence消息
             * @callback connection~onPresence
             */
            /**
             * 收到Received消息
             * @callback connection~onReceivedMessage
             */
            /**
             * 被邀请进群
             * @callback connection~onInviteMessage
             */
            /**
             * 收到已送达回执
             * @callback connection~onDeliverdMessage
             */
            /**
             * 收到已读回执
             * @callback connection~onReadMessage
             */
            /**
             * 被群管理员禁言
             * @callback connection~onMutedMessage
             */
            /**
             * 浏览器被断网时调用
             * @callback connection~onOffline
             */
            /**
             * 浏览器联网时调用
             * @callback connection~onOnline
             */
            /**
             * 建群成功后调用
             * @callback connection~onCreateGroup
             */
            this.onOpened = options.onOpened || _utils.emptyfn;
            this.onClosed = options.onClosed || _utils.emptyfn;
            this.onTextMessage = options.onTextMessage || _utils.emptyfn;
            this.onEmojiMessage = options.onEmojiMessage || _utils.emptyfn;
            this.onPictureMessage = options.onPictureMessage || _utils.emptyfn;
            this.onAudioMessage = options.onAudioMessage || _utils.emptyfn;
            this.onVideoMessage = options.onVideoMessage || _utils.emptyfn;
            this.onFileMessage = options.onFileMessage || _utils.emptyfn;
            this.onLocationMessage = options.onLocationMessage || _utils.emptyfn;
            this.onCmdMessage = options.onCmdMessage || _utils.emptyfn;
            this.onPresence = options.onPresence || _utils.emptyfn;
            this.onRoster = options.onRoster || _utils.emptyfn;
            this.onError = options.onError || _utils.emptyfn;
            this.onReceivedMessage = options.onReceivedMessage || _utils.emptyfn;
            this.onInviteMessage = options.onInviteMessage || _utils.emptyfn;
            this.onDeliverdMessage = options.onDeliveredMessage || _utils.emptyfn;
            this.onReadMessage = options.onReadMessage || _utils.emptyfn;
            this.onMutedMessage = options.onMutedMessage || _utils.emptyfn;
            this.onOffline = options.onOffline || _utils.emptyfn;
            this.onOnline = options.onOnline || _utils.emptyfn;
            this.onConfirmPop = options.onConfirmPop || _utils.emptyfn;
            this.onCreateGroup = options.onCreateGroup || _utils.emptyfn;
            //for WindowSDK start
            this.onUpdateMyGroupList = options.onUpdateMyGroupList || _utils.emptyfn;
            this.onUpdateMyRoster = options.onUpdateMyRoster || _utils.emptyfn;
            //for WindowSDK end
            this.onBlacklistUpdate = options.onBlacklistUpdate || _utils.emptyfn;

            _listenNetwork(this.onOnline, this.onOffline);
        };

        /**
         * 发送心跳
         * webrtc需要强制心跳，加个默认为false的参数 向下兼容
         * @param {Boolean} forcing - 是否强制发送
         * @private
         */
        connection.prototype.heartBeat = function (forcing) {
            if (forcing !== true) {
                forcing = false;
            }
            var me = this;
            //IE8: strophe auto switch from ws to BOSH, need heartbeat
            var isNeed = !/^ws|wss/.test(me.url) || /mobile/.test(navigator.userAgent);

            if (this.heartBeatID || !forcing && !isNeed) {
                return;
            }

            var options = {
                toJid: this.domain,
                type: 'normal'
            };
            this.heartBeatID = setInterval(function () {
                // fix: do heartbeat only when websocket
                _utils.isSupportWss && me.ping(options);
            }, this.heartBeatWait);
        };

        /**
         * @private
         */
        connection.prototype.stopHeartBeat = function () {
            if (typeof this.heartBeatID == "number") {
                this.heartBeatID = clearInterval(this.heartBeatID);
            }
        };

        /**
         * 发送接收消息回执
         * @param {Object} options -
         * @private
         */
        connection.prototype.sendReceiptsMessage = function (options) {
            var dom = $msg({
                from: this.context.jid || '',
                to: this.domain,
                id: options.id || ''
            }).c('received', {
                xmlns: 'urn:xmpp:receipts',
                id: options.id || ''
            });
            this.sendCommand(dom.tree());
        };

        /**
         * @private
         */
        connection.prototype.cacheReceiptsMessage = function (options) {
            this.sendQueue.push(options);
        };

        /**
         * @private
         */
        connection.prototype.getStrophe = function () {
            if (location.protocol != 'https:' && this.isHttpDNS) {
                //TODO: try this.xmppTotal times on fail
                var url = '';
                var host = this.xmppHosts[this.xmppIndex];
                var domain = _utils.getXmlFirstChild(host, 'domain');
                var ip = _utils.getXmlFirstChild(host, 'ip');
                if (ip) {
                    url = ip.textContent;
                    var port = _utils.getXmlFirstChild(host, 'port');
                    if (port.textContent != '80') {
                        url += ':' + port.textContent;
                    }
                } else {
                    url = domain.textContent;
                }

                if (url != '') {
                    var parter = /(.+\/\/).+(\/.+)/;
                    this.url = this.url.replace(parter, "$1" + url + "$2");
                }
            }

            var stropheConn = new Strophe.Connection(this.url, {
                isMultiLoginSessions: this.isMultiLoginSessions,
                inactivity: this.inactivity,
                maxRetries: this.maxRetries,
                pollingTime: this.pollingTime
            });
            return stropheConn;
        };

        /**
         *
         * @param data
         * @param tagName
         * @private
         */
        connection.prototype.getHostsByTag = function (data, tagName) {
            var tag = _utils.getXmlFirstChild(data, tagName);
            if (!tag) {
                console.log(tagName + ' hosts error');
                return null;
            }
            var hosts = tag.getElementsByTagName('hosts');
            if (hosts.length == 0) {
                console.log(tagName + ' hosts error2');
                return null;
            }
            return hosts[0].getElementsByTagName('host');
        };

        /**
         * @private
         */
        connection.prototype.getRestFromHttpDNS = function (options, type) {
            if (this.restIndex > this.restTotal) {
                console.log('rest hosts all tried,quit');
                return;
            }
            var url = '';
            var host = this.restHosts[this.restIndex];
            var domain = _utils.getXmlFirstChild(host, 'domain');
            var ip = _utils.getXmlFirstChild(host, 'ip');
            if (ip) {
                var port = _utils.getXmlFirstChild(host, 'port');
                url = (location.protocol === 'https:' ? 'https:' : 'http:') + '//' + ip.textContent + ':' + port.textContent;
            } else {
                url = (location.protocol === 'https:' ? 'https:' : 'http:') + '//' + domain.textContent;
            }

            if (url != '') {
                this.apiUrl = url;
                options.apiUrl = url;
            }

            if (type == 'login') {
                this.login(options);
            } else {
                this.signup(options);
            }
        };

        /**
         * @private
         */
        connection.prototype.getHttpDNS = function (options, type) {
            if (this.restHosts) {
                this.getRestFromHttpDNS(options, type);
                return;
            }
            var self = this;
            var suc = function suc(data, xhr) {
                data = new DOMParser().parseFromString(data, "text/xml").documentElement;
                //get rest ips
                var restHosts = self.getHostsByTag(data, 'rest');
                if (!restHosts) {
                    console.log('rest hosts error3');
                    return;
                }
                self.restHosts = restHosts;
                self.restTotal = restHosts.length;

                //get xmpp ips
                var xmppHosts = self.getHostsByTag(data, 'xmpp');
                if (!xmppHosts) {
                    console.log('xmpp hosts error3');
                    return;
                }
                self.xmppHosts = xmppHosts;
                self.xmppTotal = xmppHosts.length;

                self.getRestFromHttpDNS(options, type);
            };
            var error = function error(res, xhr, msg) {

                console.log('getHttpDNS error', res, msg);
                self.dnsIndex++;
                if (self.dnsIndex < self.dnsTotal) {
                    self.getHttpDNS(options, type);
                }
            };
            var options2 = {
                url: this.dnsArr[this.dnsIndex] + '/easemob/server.xml',
                dataType: 'text',
                type: 'GET',

                // url: 'http://www.easemob.com/easemob/server.xml',
                // dataType: 'xml',
                data: {app_key: encodeURIComponent(options.appKey)},
                success: suc || _utils.emptyfn,
                error: error || _utils.emptyfn
            };
            _utils.ajax(options2);
        };

        /**
         * @private
         */
        connection.prototype.signup = function (options) {
            var self = this;
            var orgName = options.orgName || '';
            var appName = options.appName || '';
            var appKey = options.appKey || '';
            var suc = options.success || EMPTYFN;
            var err = options.error || EMPTYFN;

            if (!orgName && !appName && appKey) {
                var devInfos = appKey.split('#');
                if (devInfos.length === 2) {
                    orgName = devInfos[0];
                    appName = devInfos[1];
                }
            }
            if (!orgName && !appName) {
                err({
                    type: _code.WEBIM_CONNCTION_APPKEY_NOT_ASSIGN_ERROR
                });
                return;
            }

            var error = function error(res, xhr, msg) {
                if (location.protocol != 'https:' && self.isHttpDNS) {
                    if (self.restIndex + 1 < self.restTotal) {
                        self.restIndex++;
                        self.getRestFromHttpDNS(options, 'signup');
                        return;
                    }
                }
                self.clear();
                err(res);
            };
            var https = options.https || https;
            var apiUrl = options.apiUrl;
            var restUrl = apiUrl + '/' + orgName + '/' + appName + '/users';

            var userjson = {
                username: options.username,
                password: options.password,
                nickname: options.nickname || ''
            };

            var userinfo = _utils.stringify(userjson);
            var options2 = {
                url: restUrl,
                dataType: 'json',
                data: userinfo,
                success: suc,
                error: error
            };
            _utils.ajax(options2);
        };

        /**
         * 登录
         * @param {Object} options - 用户信息
         * @param {String} options.user - 用户名
         * @param {String} options.pwd - 用户密码，跟token二选一
         * @param {String} options.accessToken - token，跟密码二选一
         * @param {String} options.appKey - Appkey
         */
        connection.prototype.open = function (options) {
            var appkey = options.appKey,
                orgName = appkey.split('#')[0],
                appName = appkey.split('#')[1];
            this.orgName = orgName;
            this.appName = appName;
            if (options.accessToken) {
                this.token = options.accessToken;
            }
            if (options.xmppURL) {
                this.url = _getXmppUrl(options.xmppURL, this.https);
            }
            if (location.protocol != 'https:' && this.isHttpDNS) {
                this.dnsIndex = 0;
                this.getHttpDNS(options, 'login');
            } else {
                this.login(options);
            }
        };

        /**
         *
         * @param options
         * @private
         */
        connection.prototype.login = function (options) {
            this.user = options.user;
            var pass = _validCheck(options, this);

            if (!pass) {
                return;
            }

            var conn = this;

            if (conn.isOpened()) {
                return;
            }

            if (options.accessToken) {
                options.access_token = options.accessToken;
                conn.context.restTokenData = options;
                _login(options, conn);
            } else {
                var apiUrl = this.apiUrl;
                var userId = this.context.userId;
                var pwd = options.pwd || '';
                var appName = this.context.appName;
                var orgName = this.context.orgName;

                var suc = function suc(data, xhr) {
                    conn.context.status = _code.STATUS_DOLOGIN_IM;
                    conn.context.restTokenData = data;
                    if (options.success) options.success(data);
                    conn.token = data.access_token;
                    _login(data, conn);
                };
                var error = function error(res, xhr, msg) {
                    if (options.error) options.error();
                    if (location.protocol != 'https:' && conn.isHttpDNS) {
                        if (conn.restIndex + 1 < conn.restTotal) {
                            conn.restIndex++;
                            conn.getRestFromHttpDNS(options, 'login');
                            return;
                        }
                    }
                    conn.clear();
                    if (res.error && res.error_description) {
                        conn.onError({
                            type: _code.WEBIM_CONNCTION_OPEN_USERGRID_ERROR,
                            data: res,
                            xhr: xhr
                        });
                    } else {
                        conn.onError({
                            type: _code.WEBIM_CONNCTION_OPEN_ERROR,
                            data: res,
                            xhr: xhr
                        });
                    }
                };

                this.context.status = _code.STATUS_DOLOGIN_USERGRID;

                var loginJson = {
                    grant_type: 'password',
                    username: userId,
                    password: pwd,
                    timestamp: +new Date()
                };
                var loginfo = _utils.stringify(loginJson);

                var options2 = {
                    url: apiUrl + '/' + orgName + '/' + appName + '/token',
                    dataType: 'json',
                    data: loginfo,
                    success: suc || _utils.emptyfn,
                    error: error || _utils.emptyfn
                };
                _utils.ajax(options2);
            }
        };

        /**
         * attach to xmpp server for BOSH
         * @private
         */
        connection.prototype.attach = function (options) {
            var pass = _validCheck(options, this);

            if (!pass) {
                return;
            }

            options = options || {};

            var accessToken = options.accessToken || '';
            if (accessToken == '') {
                this.onError({
                    type: _code.WEBIM_CONNCTION_TOKEN_NOT_ASSIGN_ERROR
                });
                return;
            }

            var sid = options.sid || '';
            if (sid === '') {
                this.onError({
                    type: _code.WEBIM_CONNCTION_SESSIONID_NOT_ASSIGN_ERROR
                });
                return;
            }

            var rid = options.rid || '';
            if (rid === '') {
                this.onError({
                    type: _code.WEBIM_CONNCTION_RID_NOT_ASSIGN_ERROR
                });
                return;
            }

            stropheConn = this.getStrophe();

            this.context.accessToken = accessToken;
            this.context.stropheConn = stropheConn;
            this.context.status = _code.STATUS_DOLOGIN_IM;

            var conn = this;
            var callback = function callback(status, msg) {
                _loginCallback(status, msg, conn);
            };

            var jid = this.context.jid;
            var wait = this.wait;
            var hold = this.hold;
            var wind = this.wind || 5;
            stropheConn.attach(jid, sid, rid, callback, wait, hold, wind);
        };

        /**
         * 关闭连接
         * @param {String} reason
         */
        connection.prototype.close = function (reason) {
            this.logOut = true;
            this.stopHeartBeat();

            var status = this.context.status;
            if (status == _code.STATUS_INIT) {
                return;
            }

            if (this.isClosed() || this.isClosing()) {
                return;
            }

            this.context.status = _code.STATUS_CLOSING;
            this.context.stropheConn.disconnect(reason);
        };

        /**
         * @private
         */
        connection.prototype.addHandler = function (handler, ns, name, type, id, from, options) {
            this.context.stropheConn.addHandler(handler, ns, name, type, id, from, options);
        };

        /**
         * @private
         */
        connection.prototype.notifyVersion = function (suc, fail) {
            var jid = stropheConn.getJid();
            this.context.jid = jid;
            var dom = $iq({
                from: jid || '',
                to: this.domain,
                type: 'result'
            }).c('query', {xmlns: 'jabber:iq:version'}).c('name').t('easemob').up().c('version').t(_version).up().c('os').t('webim');

            var suc = suc || _utils.emptyfn;
            var error = fail || this.onError;
            var failFn = function failFn(ele) {
                error({
                    type: _code.WEBIM_CONNCTION_NOTIFYVERSION_ERROR,
                    data: ele
                });
            };
            this.context.stropheConn.sendIQ(dom.tree(), suc, failFn);
            return;
        };

        /**
         * handle all types of presence message
         * @private
         */
        connection.prototype.handlePresence = function (msginfo) {
            if (this.isClosed()) {
                return;
            }
            var from = msginfo.getAttribute('from') || '';
            var to = msginfo.getAttribute('to') || '';
            var type = msginfo.getAttribute('type') || '';
            var presence_type = msginfo.getAttribute('presence_type') || '';
            var fromUser = _parseNameFromJidFn(from);
            var toUser = _parseNameFromJidFn(to);
            var isCreate = false;
            var isMemberJoin = false;
            var isDecline = false;
            var isApply = false;
            var info = {
                from: fromUser,
                to: toUser,
                fromJid: from,
                toJid: to,
                type: type,
                chatroom: msginfo.getElementsByTagName('roomtype').length ? true : false
            };

            var showTags = msginfo.getElementsByTagName('show');
            if (showTags && showTags.length > 0) {
                var showTag = showTags[0];
                info.show = Strophe.getText(showTag);
            }
            var statusTags = msginfo.getElementsByTagName('status');
            if (statusTags && statusTags.length > 0) {
                var statusTag = statusTags[0];
                info.status = Strophe.getText(statusTag);
                info.code = statusTag.getAttribute('code');
            }

            var priorityTags = msginfo.getElementsByTagName('priority');
            if (priorityTags && priorityTags.length > 0) {
                var priorityTag = priorityTags[0];
                info.priority = Strophe.getText(priorityTag);
            }

            var error = msginfo.getElementsByTagName('error');
            if (error && error.length > 0) {
                var error = error[0];
                info.error = {
                    code: error.getAttribute('code')
                };
            }

            var destroy = msginfo.getElementsByTagName('destroy');
            if (destroy && destroy.length > 0) {
                var destroy = destroy[0];
                info.destroy = true;

                var reason = destroy.getElementsByTagName('reason');
                if (reason && reason.length > 0) {
                    info.reason = Strophe.getText(reason[0]);
                }
            }

            var members = msginfo.getElementsByTagName('item');
            if (members && members.length > 0) {
                var member = members[0];
                var role = member.getAttribute('role');
                var jid = member.getAttribute('jid');
                var affiliation = member.getAttribute('affiliation');
                // dismissed by group
                if (role == 'none' && jid) {
                    var kickedMember = _parseNameFromJidFn(jid);
                    var actor = member.getElementsByTagName('actor')[0];
                    var actorNick = actor.getAttribute('nick');
                    info.actor = actorNick;
                    info.kicked = kickedMember;
                }
                // Service Acknowledges Room Creation `createGroupACK`
                if (role == 'moderator' && info.code == '201') {
                    if (affiliation === 'owner') {
                        info.type = 'createGroupACK';
                        isCreate = true;
                    }
                    // else
                    //     info.type = 'joinPublicGroupSuccess';
                }
            }

            var x = msginfo.getElementsByTagName('x');
            if (x && x.length > 0) {
                // 加群申请
                var apply = x[0].getElementsByTagName('apply');
                // 加群成功
                var accept = x[0].getElementsByTagName('accept');
                // 同意加群后用户进群通知
                var item = x[0].getElementsByTagName('item');
                // 加群被拒绝
                var decline = x[0].getElementsByTagName('decline');
                // 被设为管理员
                var addAdmin = x[0].getElementsByTagName('add_admin');
                // 被取消管理员
                var removeAdmin = x[0].getElementsByTagName('remove_admin');
                // 被禁言
                var addMute = x[0].getElementsByTagName('add_mute');
                // 取消禁言
                var removeMute = x[0].getElementsByTagName('remove_mute');

                if (apply && apply.length > 0) {
                    isApply = true;
                    info.toNick = apply[0].getAttribute('toNick');
                    info.type = 'joinGroupNotifications';
                    var groupJid = apply[0].getAttribute('to');
                    var gid = groupJid.split('@')[0].split('_');
                    gid = gid[gid.length - 1];
                    info.gid = gid;
                } else if (accept && accept.length > 0) {
                    info.type = 'joinPublicGroupSuccess';
                } else if (item && item.length > 0) {
                    var affiliation = item[0].getAttribute('affiliation'),
                        role = item[0].getAttribute('role');
                    if (affiliation == 'member' || role == 'participant') {
                        isMemberJoin = true;
                        info.mid = info.fromJid.split('/');
                        info.mid = info.mid[info.mid.length - 1];
                        info.type = 'memberJoinPublicGroupSuccess';
                        var roomtype = msginfo.getElementsByTagName('roomtype');
                        if (roomtype && roomtype.length > 0) {
                            var type = roomtype[0].getAttribute('type');
                            if (type == 'chatroom') {
                                info.type = 'memberJoinChatRoomSuccess';
                            }
                        }
                    }
                } else if (decline && decline.length) {
                    isDecline = true;
                    var gid = decline[0].getAttribute("fromNick");
                    var owner = _parseNameFromJidFn(decline[0].getAttribute("from"));
                    info.type = "joinPublicGroupDeclined";
                    info.owner = owner;
                    info.gid = gid;
                } else if (addAdmin && addAdmin.length > 0) {
                    var gid = _parseNameFromJidFn(addAdmin[0].getAttribute('mucjid'));
                    var owner = _parseNameFromJidFn(addAdmin[0].getAttribute('from'));
                    info.owner = owner;
                    info.gid = gid;
                    info.type = "addAdmin";
                } else if (removeAdmin && removeAdmin.length > 0) {
                    var gid = _parseNameFromJidFn(removeAdmin[0].getAttribute('mucjid'));
                    var owner = _parseNameFromJidFn(removeAdmin[0].getAttribute('from'));
                    info.owner = owner;
                    info.gid = gid;
                    info.type = "removeAdmin";
                } else if (addMute && addMute.length > 0) {
                    var gid = _parseNameFromJidFn(addMute[0].getAttribute('mucjid'));
                    var owner = _parseNameFromJidFn(addMute[0].getAttribute('from'));
                    info.owner = owner;
                    info.gid = gid;
                    info.type = "addMute";
                } else if (removeMute && removeMute.length > 0) {
                    var gid = _parseNameFromJidFn(removeMute[0].getAttribute('mucjid'));
                    var owner = _parseNameFromJidFn(removeMute[0].getAttribute('from'));
                    info.owner = owner;
                    info.gid = gid;
                    info.type = "removeMute";
                }
            }

            if (info.chatroom) {
                // diff the
                info.presence_type = presence_type;
                info.original_type = info.type;
                var reflectUser = from.slice(from.lastIndexOf('/') + 1);

                if (reflectUser === this.context.userId) {
                    if (info.type === '' && !info.code) {
                        info.type = 'joinChatRoomSuccess';
                    } else if (presence_type === 'unavailable' || info.type === 'unavailable') {
                        if (!info.status) {
                            // logout successfully.
                            info.type = 'leaveChatRoom';
                        } else if (info.code == 110) {
                            // logout or dismissied by admin.
                            info.type = 'leaveChatRoom';
                        } else if (info.error && info.error.code == 406) {
                            // The chat room is full.
                            info.type = 'reachChatRoomCapacity';
                        }
                    }
                }
            } else {
                info.presence_type = presence_type;
                info.original_type = type;

                if (/subscribe/.test(info.type)) {
                    //subscribe | subscribed | unsubscribe | unsubscribed
                } else if (type == "" && !info.status && !info.error && !isCreate && !isApply && !isMemberJoin && !isDecline) {
                    // info.type = 'joinPublicGroupSuccess';
                } else if (presence_type === 'unavailable' || type === 'unavailable') {
                    // There is no roomtype when a chat room is deleted.
                    if (info.destroy) {
                        // Group or Chat room Deleted.
                        info.type = 'deleteGroupChat';
                    } else if (info.code == 307 || info.code == 321) {
                        // Dismissed by group.
                        var nick = msginfo.getAttribute('nick');
                        if (!nick) info.type = 'leaveGroup'; else info.type = 'removedFromGroup';
                    }
                }
            }
            this.onPresence(info, msginfo);
        };

        /**
         * @private
         */
        connection.prototype.handlePing = function (e) {
            if (this.isClosed()) {
                return;
            }
            var id = e.getAttribute('id');
            var from = e.getAttribute('from');
            var to = e.getAttribute('to');
            var dom = $iq({
                from: to,
                to: from,
                id: id,
                type: 'result'
            });
            this.sendCommand(dom.tree());
        };

        /**
         * @private
         */
        connection.prototype.handleIq = function (iq) {
            return true;
        };

        /**
         * @private
         */
        connection.prototype.handleIqPrivacy = function (msginfo) {
            var list = msginfo.getElementsByTagName('list');
            if (list.length == 0) {
                return;
            }
            this.getBlacklist();
        };

        /**
         * @private
         */
        connection.prototype.handleIqRoster = function (e) {
            var id = e.getAttribute('id');
            var from = e.getAttribute('from') || '';
            var name = _parseNameFromJidFn(from);
            var curJid = this.context.jid;
            var curUser = this.context.userId;

            var iqresult = $iq({type: 'result', id: id, from: curJid});
            this.sendCommand(iqresult.tree());

            var msgBodies = e.getElementsByTagName('query');
            if (msgBodies && msgBodies.length > 0) {
                var queryTag = msgBodies[0];
                var rouster = _parseFriend(queryTag, this, from);
                this.onRoster(rouster);
            }
            return true;
        };

        /**
         * @private
         */
        connection.prototype.handleMessage = function (msginfo) {
            var self = this;
            if (this.isClosed()) {
                return;
            }

            var id = msginfo.getAttribute('id') || '';

            // cache ack into sendQueue first , handelSendQueue will do the send thing with the speed of  5/s
            this.cacheReceiptsMessage({
                id: id
            });
            var parseMsgData = _parseResponseMessage(msginfo);
            if (parseMsgData.errorMsg) {
                this.handlePresence(msginfo);
                return;
            }
            // send error
            var error = msginfo.getElementsByTagName('error');
            var errorCode = '';
            var errorText = '';
            var errorBool = false;
            if (error.length > 0) {
                errorBool = true;
                errorCode = error[0].getAttribute('code');
                var textDOM = error[0].getElementsByTagName('text');
                errorText = textDOM[0].textContent || textDOM[0].text;
            }

            var msgDatas = parseMsgData.data;
            for (var i in msgDatas) {
                if (!msgDatas.hasOwnProperty(i)) {
                    continue;
                }
                var msg = msgDatas[i];
                if (!msg.from || !msg.to) {
                    continue;
                }

                var from = (msg.from + '').toLowerCase();
                var too = (msg.to + '').toLowerCase();
                var extmsg = msg.ext || {};
                var chattype = '';
                var typeEl = msginfo.getElementsByTagName('roomtype');
                if (typeEl.length) {
                    chattype = typeEl[0].getAttribute('type') || 'chat';
                } else {
                    chattype = msginfo.getAttribute('type') || 'chat';
                }

                var msgBodies = msg.bodies;
                if (!msgBodies || msgBodies.length == 0) {
                    continue;
                }
                var msgBody = msg.bodies[0];
                var type = msgBody.type;

                try {
                    switch (type) {
                        case 'txt':
                            var receiveMsg = msgBody.msg;
                            var sourceMsg = _.clone(receiveMsg);
                            /*
	                     if (self.encrypt.type === 'base64') {
	                     receiveMsg = atob(receiveMsg);
	                     } else if (self.encrypt.type === 'aes') {
	                     var key = CryptoJS.enc.Utf8.parse(self.encrypt.key);
	                     var iv = CryptoJS.enc.Utf8.parse(self.encrypt.iv);
	                     var mode = self.encrypt.mode.toLowerCase();
	                     var option = {};
	                     if (mode === 'cbc') {
	                     option = {
	                     iv: iv,
	                     mode: CryptoJS.mode.CBC,
	                     padding: CryptoJS.pad.Pkcs7
	                     };
	                     } else if (mode === 'ebc') {
	                     option = {
	                     mode: CryptoJS.mode.ECB,
	                     padding: CryptoJS.pad.Pkcs7
	                     }
	                     }
	                     var encryptedBase64Str = receiveMsg;
	                     var decryptedData = CryptoJS.AES.decrypt(encryptedBase64Str, key, option);
	                     var decryptedStr = decryptedData.toString(CryptoJS.enc.Utf8);
	                     receiveMsg = decryptedStr;
	                     }
	                     */
                            receiveMsg = self.decrypt(receiveMsg);
                            var emojibody = _utils.parseTextMessage(receiveMsg, WebIM.Emoji);
                            if (emojibody.isemoji) {
                                var msg = {
                                    id: id,
                                    type: chattype,
                                    from: from,
                                    to: too,
                                    delay: parseMsgData.delayTimeStamp,
                                    data: emojibody.body,
                                    ext: extmsg,
                                    sourceMsg: sourceMsg
                                };
                                !msg.delay && delete msg.delay;
                                msg.error = errorBool;
                                msg.errorText = errorText;
                                msg.errorCode = errorCode;
                                this.onEmojiMessage(msg);
                            } else {
                                var msg = {
                                    id: id,
                                    type: chattype,
                                    from: from,
                                    to: too,
                                    delay: parseMsgData.delayTimeStamp,
                                    data: receiveMsg,
                                    ext: extmsg,
                                    sourceMsg: sourceMsg
                                };
                                !msg.delay && delete msg.delay;
                                msg.error = errorBool;
                                msg.errorText = errorText;
                                msg.errorCode = errorCode;
                                this.onTextMessage(msg);
                            }
                            break;
                        case 'img':
                            var rwidth = 0;
                            var rheight = 0;
                            if (msgBody.size) {
                                rwidth = msgBody.size.width;
                                rheight = msgBody.size.height;
                            }
                            var msg = {
                                id: id,
                                type: chattype,
                                from: from,
                                to: too,

                                url: location.protocol != 'https:' && self.isHttpDNS ? self.apiUrl + msgBody.url.substr(msgBody.url.indexOf("/", 9)) : msgBody.url,
                                secret: msgBody.secret,
                                filename: msgBody.filename,
                                thumb: msgBody.thumb,
                                thumb_secret: msgBody.thumb_secret,
                                file_length: msgBody.file_length || '',
                                width: rwidth,
                                height: rheight,
                                filetype: msgBody.filetype || '',
                                accessToken: this.context.accessToken || '',
                                ext: extmsg,
                                delay: parseMsgData.delayTimeStamp
                            };
                            !msg.delay && delete msg.delay;
                            msg.error = errorBool;
                            msg.errorText = errorText;
                            msg.errorCode = errorCode;
                            this.onPictureMessage(msg);
                            break;
                        case 'audio':
                            var msg = {
                                id: id,
                                type: chattype,
                                from: from,
                                to: too,

                                url: location.protocol != 'https:' && self.isHttpDNS ? self.apiUrl + msgBody.url.substr(msgBody.url.indexOf("/", 9)) : msgBody.url,
                                secret: msgBody.secret,
                                filename: msgBody.filename,
                                length: msgBody.length || '',
                                file_length: msgBody.file_length || '',
                                filetype: msgBody.filetype || '',
                                accessToken: this.context.accessToken || '',
                                ext: extmsg,
                                delay: parseMsgData.delayTimeStamp
                            };
                            !msg.delay && delete msg.delay;
                            msg.error = errorBool;
                            msg.errorText = errorText;
                            msg.errorCode = errorCode;
                            this.onAudioMessage(msg);
                            break;
                        case 'file':
                            var msg = {
                                id: id,
                                type: chattype,
                                from: from,
                                to: too,

                                url: location.protocol != 'https:' && self.isHttpDNS ? self.apiUrl + msgBody.url.substr(msgBody.url.indexOf("/", 9)) : msgBody.url,
                                secret: msgBody.secret,
                                filename: msgBody.filename,
                                file_length: msgBody.file_length,
                                accessToken: this.context.accessToken || '',
                                ext: extmsg,
                                delay: parseMsgData.delayTimeStamp
                            };
                            !msg.delay && delete msg.delay;
                            msg.error = errorBool;
                            msg.errorText = errorText;
                            msg.errorCode = errorCode;
                            this.onFileMessage(msg);
                            break;
                        case 'loc':
                            var msg = {
                                id: id,
                                type: chattype,
                                from: from,
                                to: too,
                                addr: msgBody.addr,
                                lat: msgBody.lat,
                                lng: msgBody.lng,
                                ext: extmsg,
                                delay: parseMsgData.delayTimeStamp
                            };
                            !msg.delay && delete msg.delay;
                            msg.error = errorBool;
                            msg.errorText = errorText;
                            msg.errorCode = errorCode;
                            this.onLocationMessage(msg);
                            break;
                        case 'video':
                            var msg = {
                                id: id,
                                type: chattype,
                                from: from,
                                to: too,

                                url: location.protocol != 'https:' && self.isHttpDNS ? self.apiUrl + msgBody.url.substr(msgBody.url.indexOf("/", 9)) : msgBody.url,
                                secret: msgBody.secret,
                                filename: msgBody.filename,
                                file_length: msgBody.file_length,
                                accessToken: this.context.accessToken || '',
                                ext: extmsg,
                                delay: parseMsgData.delayTimeStamp
                            };
                            !msg.delay && delete msg.delay;
                            msg.error = errorBool;
                            msg.errorText = errorText;
                            msg.errorCode = errorCode;
                            this.onVideoMessage(msg);
                            break;
                        case 'cmd':
                            var msg = {
                                id: id,
                                from: from,
                                to: too,
                                action: msgBody.action,
                                ext: extmsg,
                                delay: parseMsgData.delayTimeStamp
                            };
                            !msg.delay && delete msg.delay;
                            msg.error = errorBool;
                            msg.errorText = errorText;
                            msg.errorCode = errorCode;
                            this.onCmdMessage(msg);
                            break;
                    }
                    ;
                    if (self.delivery) {
                        var msgId = self.getUniqueId();
                        var bodyId = msg.id;
                        var deliverMessage = new WebIM.message('delivery', msgId);
                        deliverMessage.set({
                            id: bodyId,
                            to: msg.from
                        });
                        self.send(deliverMessage.body);
                    }
                } catch (e) {
                    this.onError({
                        type: _code.WEBIM_CONNCTION_CALLBACK_INNER_ERROR,
                        data: e
                    });
                }
            }
        };

        /**
         * @private
         */
        connection.prototype.handleDeliveredMessage = function (message) {
            var id = message.id;
            var body = message.getElementsByTagName('body');
            var mid = 0;
            mid = body[0].innerHTML;
            var msg = {
                mid: mid
            };
            this.onDeliverdMessage(msg);
            this.sendReceiptsMessage({
                id: id
            });
        };

        /**
         * @private
         */
        connection.prototype.handleAckedMessage = function (message) {
            var id = message.id;
            var body = message.getElementsByTagName('body');
            var mid = 0;
            mid = body[0].innerHTML;
            var msg = {
                mid: mid
            };
            this.onReadMessage(msg);
            this.sendReceiptsMessage({
                id: id
            });
        };

        /**
         * @private
         */
        connection.prototype.handleReceivedMessage = function (message) {
            try {
                var received = message.getElementsByTagName("received");
                var mid = received[0].getAttribute('mid');
                var body = message.getElementsByTagName("body");
                var id = body[0].innerHTML;
                var msg = {
                    mid: mid,
                    id: id
                };
                this.onReceivedMessage(msg);
            } catch (e) {
                this.onError({
                    type: _code.WEBIM_CONNCTION_CALLBACK_INNER_ERROR,
                    data: e
                });
            }

            var rcv = message.getElementsByTagName('received'),
                id,
                mid;

            if (rcv.length > 0) {
                if (rcv[0].childNodes && rcv[0].childNodes.length > 0) {
                    id = rcv[0].childNodes[0].nodeValue;
                } else {
                    id = rcv[0].innerHTML || rcv[0].innerText;
                }
                mid = rcv[0].getAttribute('mid');
            }

            if (_msgHash[id]) {
                try {
                    _msgHash[id].msg.success instanceof Function && _msgHash[id].msg.success(id, mid);
                } catch (e) {
                    this.onError({
                        type: _code.WEBIM_CONNCTION_CALLBACK_INNER_ERROR,
                        data: e
                    });
                }
                delete _msgHash[id];
            }
        };

        /**
         * @private
         */
        connection.prototype.handleInviteMessage = function (message) {
            var form = null;
            var invitemsg = message.getElementsByTagName('invite');
            var reasonDom = message.getElementsByTagName('reason')[0];
            var reasonMsg = reasonDom.textContent;
            var id = message.getAttribute('id') || '';
            this.sendReceiptsMessage({
                id: id
            });

            if (invitemsg && invitemsg.length > 0) {
                var fromJid = invitemsg[0].getAttribute('from');
                form = _parseNameFromJidFn(fromJid);
            }
            var xmsg = message.getElementsByTagName('x');
            var roomid = null;
            if (xmsg && xmsg.length > 0) {
                for (var i = 0; i < xmsg.length; i++) {
                    if ('jabber:x:conference' === xmsg[i].namespaceURI) {
                        var roomjid = xmsg[i].getAttribute('jid');
                        roomid = _parseNameFromJidFn(roomjid);
                    }
                }
            }
            this.onInviteMessage({
                type: 'invite',
                from: form,
                roomid: roomid,
                reason: reasonMsg
            });
        };

        /**
         * @private
         */
        connection.prototype.handleMutedMessage = function (message) {
            var id = message.id;
            this.onMutedMessage({
                mid: id
            });
        };

        /**
         * @private
         */
        connection.prototype.sendCommand = function (dom, id) {
            if (this.isOpened()) {
                this.context.stropheConn.send(dom);
            } else {
                this.unSendMsgArr.push(dom);
                if (!this.offLineSendConnecting && !this.logOut) {
                    this.offLineSendConnecting = true;
                    this.reconnect();
                }
                this.onError({
                    type: _code.WEBIM_CONNCTION_DISCONNECTED,
                    reconnect: true
                });
            }
        };

        /**
         * 随机生成一个id用于消息id
         * @param {String} [prefix=WEBIM_] - 前缀
         * @returns {String} 唯一的id
         */
        connection.prototype.getUniqueId = function (prefix) {
            // fix: too frequently msg sending will make same id
            if (this.autoIncrement) {
                this.autoIncrement++;
            } else {
                this.autoIncrement = 1;
            }
            var cdate = new Date();
            var offdate = new Date(2010, 1, 1);
            var offset = cdate.getTime() - offdate.getTime();
            var hexd = parseFloat(offset).toString(16) + this.autoIncrement;

            if (typeof prefix === 'string' || typeof prefix === 'number') {
                return prefix + '_' + hexd;
            } else {
                return 'WEBIM_' + hexd;
            }
        };

        /**
         * send message
         * @param {Object} messageSource - 由 Class Message 生成
         */
        connection.prototype.send = function (messageSource) {
            var self = this;
            var message = messageSource;
            if (message.type === 'txt') {
                if (this.encrypt.type === 'base64') {
                    message = _.clone(messageSource);
                    message.msg = btoa(message.msg);
                } else if (this.encrypt.type === 'aes') {
                    message = _.clone(messageSource);
                    var key = CryptoJS.enc.Utf8.parse(this.encrypt.key);
                    var iv = CryptoJS.enc.Utf8.parse(this.encrypt.iv);
                    var mode = this.encrypt.mode.toLowerCase();
                    var option = {};
                    if (mode === 'cbc') {
                        option = {
                            iv: iv,
                            mode: CryptoJS.mode.CBC,
                            padding: CryptoJS.pad.Pkcs7
                        };
                    } else if (mode === 'ebc') {
                        option = {
                            mode: CryptoJS.mode.ECB,
                            padding: CryptoJS.pad.Pkcs7
                        };
                    }
                    var encryptedData = CryptoJS.AES.encrypt(message.msg, key, option);

                    message.msg = encryptedData.toString();
                }
            }
            if (this.isWindowSDK) {
                WebIM.doQuery('{"type":"sendMessage","to":"' + message.to + '","message_type":"' + message.type + '","msg":"' + encodeURI(message.msg) + '","chatType":"' + message.chatType + '"}', function (response) {
                }, function (code, msg) {
                    var message = {
                        data: {
                            data: "send"
                        },
                        type: _code.WEBIM_MESSAGE_SED_ERROR
                    };
                    self.onError(message);
                });
            } else {
                if (Object.prototype.toString.call(message) === '[object Object]') {
                    var appKey = this.context.appKey || '';
                    var toJid = appKey + '_' + message.to + '@' + this.domain;

                    if (message.group) {
                        toJid = appKey + '_' + message.to + '@conference.' + this.domain;
                    }
                    if (message.resource) {
                        toJid = toJid + '/' + message.resource;
                    }

                    message.toJid = toJid;
                    message.id = message.id || this.getUniqueId();
                    _msgHash[message.id] = new _message(message);
                    _msgHash[message.id].send(this);
                } else if (typeof message === 'string') {
                    _msgHash[message] && _msgHash[message].send(this);
                }
            }
        };

        /**
         * 添加联系人，已废弃不用
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.addRoster = function (options) {
            var jid = _getJid(options, this);
            var name = options.name || '';
            var groups = options.groups || '';

            var iq = $iq({type: 'set'});
            iq.c('query', {xmlns: 'jabber:iq:roster'});
            iq.c('item', {jid: jid, name: name});

            if (groups) {
                for (var i = 0; i < groups.length; i++) {
                    iq.c('group').t(groups[i]).up();
                }
            }
            var suc = options.success || _utils.emptyfn;
            var error = options.error || _utils.emptyfn;
            this.context.stropheConn.sendIQ(iq.tree(), suc, error);
        };

        /**
         * 删除联系人
         *
         * @param {Object} options
         * @param {String} options.to - 想要删除的联系人ID
         * @param {Function} options.success - 成功回调，在这里面调用connection.unsubscribed才能真正删除联系人
         * @fires connection#unsubscribed
         */
        connection.prototype.removeRoster = function (options) {
            var jid = _getJid(options, this);
            var iq = $iq({type: 'set'}).c('query', {xmlns: 'jabber:iq:roster'}).c('item', {
                jid: jid,
                subscription: 'remove'
            });
            var suc = options.success || _utils.emptyfn;
            var error = options.error || _utils.emptyfn;
            this.context.stropheConn.sendIQ(iq, suc, error);
        };

        /**
         * 获取联系人
         * @param {Object} options
         * @param {Function} options.success - 获取好友列表成功
         */
        connection.prototype.getRoster = function (options) {
            var dom = $iq({
                type: 'get'
            }).c('query', {xmlns: 'jabber:iq:roster'});

            var options = options || {};
            var suc = options.success || this.onRoster;
            var completeFn = function completeFn(ele) {
                var rouster = [];
                var msgBodies = ele.getElementsByTagName('query');
                if (msgBodies && msgBodies.length > 0) {
                    var queryTag = msgBodies[0];
                    rouster = _parseFriend(queryTag);
                }
                suc(rouster, ele);
            };
            var error = options.error || this.onError;
            var failFn = function failFn(ele) {
                error({
                    type: _code.WEBIM_CONNCTION_GETROSTER_ERROR,
                    data: ele
                });
            };
            if (this.isOpened()) {
                this.context.stropheConn.sendIQ(dom.tree(), completeFn, failFn);
            } else {
                error({
                    type: _code.WEBIM_CONNCTION_DISCONNECTED
                });
            }
        };

        /**
         * 订阅和反向订阅
         * @example
         *
         * A订阅B（A添加B为好友）
         * A执行：
         *  conn.subscribe({
	                to: 'B',
	                message: 'Hello~'
	            });
         B的监听函数onPresence参数message.type == subscribe监听到有人订阅他
         B执行：
         conn.subscribed({
	                to: 'A',
	                message: '[resp:true]'
	          });
         同意A的订阅请求
         B继续执行：
         conn.subscribe({
	                to: 'A',
	                message: '[resp:true]'
	            });
         反向订阅A，这样才算双方添加好友成功。
         若B拒绝A的订阅请求，只需执行：
         conn.unsubscribed({
	                        to: 'A',
	                        message: 'I don't want to be subscribed'
	                    });
         另外，在监听函数onPresence参数message.type == "subscribe"这个case中，加一句
         if (message && message.status === '[resp:true]') {
	            return;
	        }
         否则会进入死循环
         *
         * @param {Object} options - 想要订阅的联系人信息
         * @param {String} options.to - 想要订阅的联系人ID
         * @param {String} options.message - 发送给想要订阅的联系人的验证消息
         */
        connection.prototype.subscribe = function (options) {
            var jid = _getJid(options, this);
            var pres = $pres({to: jid, type: 'subscribe'});
            if (options.message) {
                pres.c('status').t(options.message).up();
            }
            if (options.nick) {
                pres.c('nick', {'xmlns': 'http://jabber.org/protocol/nick'}).t(options.nick);
            }
            this.sendCommand(pres.tree());
        };

        /**
         * 被订阅后确认同意被订阅
         * @param {Object} options - 订阅人的信息
         * @param {String} options.to - 订阅人的ID
         * @param {String} options.message=[resp:true] - 默认为[resp:true]，后续将去掉该参数
         */
        connection.prototype.subscribed = function (options) {
            var message = '[resp:true]';
            var jid = _getJid(options, this);
            var pres = $pres({to: jid, type: 'subscribed'});

            if (options.message) {
                pres.c('status').t(options.message).up();
            }
            this.sendCommand(pres.tree());
        };

        /**
         * 取消订阅成功，废弃不用
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.unsubscribe = function (options) {
            var jid = _getJid(options, this);
            var pres = $pres({to: jid, type: 'unsubscribe'});

            if (options.message) {
                pres.c('status').t(options.message);
            }
            this.sendCommand(pres.tree());
        };

        /**
         * 拒绝对方的订阅请求
         * @function connection#event:unsubscribed
         * @param {Object} options -
         * @param {String} options.to - 订阅人的ID
         */
        connection.prototype.unsubscribed = function (options) {
            var jid = _getJid(options, this);
            var pres = $pres({to: jid, type: 'unsubscribed'});

            if (options.message) {
                pres.c('status').t(options.message).up();
            }
            this.sendCommand(pres.tree());
        };

        /**
         * 加入公开群组
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.joinPublicGroup = function (options) {
            var roomJid = this.context.appKey + '_' + options.roomId + '@conference.' + this.domain;
            var room_nick = roomJid + '/' + this.context.userId;
            var suc = options.success || _utils.emptyfn;
            var err = options.error || _utils.emptyfn;
            var errorFn = function errorFn(ele) {
                err({
                    type: _code.WEBIM_CONNCTION_JOINROOM_ERROR,
                    data: ele
                });
            };
            var iq = $pres({
                from: this.context.jid,
                to: room_nick
            }).c('x', {xmlns: Strophe.NS.MUC});

            this.context.stropheConn.sendIQ(iq.tree(), suc, errorFn);
        };

        /**
         * 获取聊天室列表
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.listRooms = function (options) {
            var iq = $iq({
                to: options.server || 'conference.' + this.domain,
                from: this.context.jid,
                type: 'get'
            }).c('query', {xmlns: Strophe.NS.DISCO_ITEMS});

            var suc = options.success || _utils.emptyfn;
            var error = options.error || this.onError;
            var completeFn = function completeFn(result) {
                var rooms = [];
                rooms = _parseRoom(result);
                try {
                    suc(rooms);
                } catch (e) {
                    error({
                        type: _code.WEBIM_CONNCTION_GETROOM_ERROR,
                        data: e
                    });
                }
            };
            var err = options.error || _utils.emptyfn;
            var errorFn = function errorFn(ele) {
                err({
                    type: _code.WEBIM_CONNCTION_GETROOM_ERROR,
                    data: ele
                });
            };
            this.context.stropheConn.sendIQ(iq.tree(), completeFn, errorFn);
        };

        /**
         * 获取群组成员列表
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.queryRoomMember = function (options) {
            var domain = this.domain;
            var members = [];
            var iq = $iq({
                to: this.context.appKey + '_' + options.roomId + '@conference.' + this.domain,
                type: 'get'
            }).c('query', {xmlns: Strophe.NS.MUC + '#admin'}).c('item', {affiliation: 'member'});

            var suc = options.success || _utils.emptyfn;
            var completeFn = function completeFn(result) {
                var items = result.getElementsByTagName('item');

                if (items) {
                    for (var i = 0; i < items.length; i++) {
                        var item = items[i];
                        var mem = {
                            jid: item.getAttribute('jid'),
                            affiliation: 'member'
                        };
                        members.push(mem);
                    }
                }
                suc(members);
            };
            var err = options.error || _utils.emptyfn;
            var errorFn = function errorFn(ele) {
                err({
                    type: _code.WEBIM_CONNCTION_GETROOMMEMBER_ERROR,
                    data: ele
                });
            };
            this.context.stropheConn.sendIQ(iq.tree(), completeFn, errorFn);
        };

        /**
         * 获取群组信息
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.queryRoomInfo = function (options) {
            console.log('QueryRoomInfo');
            var domain = this.domain;
            var iq = $iq({
                to: this.context.appKey + '_' + options.roomId + '@conference.' + domain,
                type: 'get'
            }).c('query', {xmlns: Strophe.NS.DISCO_INFO});

            var suc = options.success || _utils.emptyfn;
            var members = [];

            var completeFn = function completeFn(result) {
                var settings = '';
                var features = result.getElementsByTagName('feature');
                if (features) {
                    settings = features[1].getAttribute('var') + '|' + features[3].getAttribute('var') + '|' + features[4].getAttribute('var');
                }
                switch (settings) {
                    case 'muc_public|muc_membersonly|muc_notallowinvites':
                        settings = 'PUBLIC_JOIN_APPROVAL';
                        break;
                    case 'muc_public|muc_open|muc_notallowinvites':
                        settings = 'PUBLIC_JOIN_OPEN';
                        break;
                    case 'muc_hidden|muc_membersonly|muc_allowinvites':
                        settings = 'PRIVATE_MEMBER_INVITE';
                        break;
                    case 'muc_hidden|muc_membersonly|muc_notallowinvites':
                        settings = 'PRIVATE_OWNER_INVITE';
                        break;
                }
                var owner = '';
                var fields = result.getElementsByTagName('field');
                var fieldValues = {};
                if (fields) {
                    for (var i = 0; i < fields.length; i++) {
                        var field = fields[i];
                        var fieldVar = field.getAttribute('var');
                        var fieldSimplify = fieldVar.split('_')[1];
                        switch (fieldVar) {
                            case 'muc#roominfo_occupants':
                            case 'muc#roominfo_maxusers':
                            case 'muc#roominfo_affiliations':
                            case 'muc#roominfo_description':
                                fieldValues[fieldSimplify] = field.textContent || field.text || '';
                                break;
                            case 'muc#roominfo_owner':
                                var mem = {
                                    jid: (field.textContent || field.text) + '@' + domain,
                                    affiliation: 'owner'
                                };
                                members.push(mem);
                                fieldValues[fieldSimplify] = field.textContent || field.text;
                                break;
                        }

                        // if (field.getAttribute('label') === 'owner') {
                        //     var mem = {
                        //         jid: (field.textContent || field.text) + '@' + domain
                        //         , affiliation: 'owner'
                        //     };
                        //     members.push(mem);
                        //     break;
                        // }
                    }
                    fieldValues['name'] = result.getElementsByTagName('identity')[0].getAttribute('name');
                }
                suc(settings, members, fieldValues);
            };
            var err = options.error || _utils.emptyfn;
            var errorFn = function errorFn(ele) {
                err({
                    type: _code.WEBIM_CONNCTION_GETROOMINFO_ERROR,
                    data: ele
                });
            };
            this.context.stropheConn.sendIQ(iq.tree(), completeFn, errorFn);
        };

        /**
         * 获取聊天室管理员
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.queryRoomOccupants = function (options) {
            var suc = options.success || _utils.emptyfn;
            var completeFn = function completeFn(result) {
                var occupants = [];
                occupants = _parseRoomOccupants(result);
                suc(occupants);
            };
            var err = options.error || _utils.emptyfn;
            var errorFn = function errorFn(ele) {
                err({
                    type: _code.WEBIM_CONNCTION_GETROOMOCCUPANTS_ERROR,
                    data: ele
                });
            };
            var attrs = {
                xmlns: Strophe.NS.DISCO_ITEMS
            };
            var info = $iq({
                from: this.context.jid,
                to: this.context.appKey + '_' + options.roomId + '@conference.' + this.domain,
                type: 'get'
            }).c('query', attrs);
            this.context.stropheConn.sendIQ(info.tree(), completeFn, errorFn);
        };

        /**
         *
         * @deprecated
         * @private
         */
        connection.prototype.setUserSig = function (desc) {
            var dom = $pres({xmlns: 'jabber:client'});
            desc = desc || '';
            dom.c('status').t(desc);
            this.sendCommand(dom.tree());
        };

        /**
         *
         * @private
         */
        connection.prototype.setPresence = function (type, status) {
            var dom = $pres({xmlns: 'jabber:client'});
            if (type) {
                if (status) {
                    dom.c('show').t(type);
                    dom.up().c('status').t(status);
                } else {
                    dom.c('show').t(type);
                }
            }
            this.sendCommand(dom.tree());
        };

        /**
         * @private
         *
         */
        connection.prototype.getPresence = function () {
            var dom = $pres({xmlns: 'jabber:client'});
            var conn = this;
            this.sendCommand(dom.tree());
        };

        /**
         * @private
         *
         */
        connection.prototype.ping = function (options) {
            var options = options || {};
            var jid = _getJid(options, this);

            var dom = $iq({
                from: this.context.jid || '',
                to: jid,
                type: 'get'
            }).c('ping', {xmlns: 'urn:xmpp:ping'});

            var suc = options.success || _utils.emptyfn;
            var error = options.error || this.onError;
            var failFn = function failFn(ele) {
                error({
                    type: _code.WEBIM_CONNCTION_PING_ERROR,
                    data: ele
                });
            };
            if (this.isOpened()) {
                this.context.stropheConn.sendIQ(dom.tree(), suc, failFn);
            } else {
                error({
                    type: _code.WEBIM_CONNCTION_DISCONNECTED
                });
            }
            return;
        };

        /**
         * @private
         *
         */
        connection.prototype.isOpened = function () {
            return this.context.status == _code.STATUS_OPENED;
        };

        /**
         * @private
         *
         */
        connection.prototype.isOpening = function () {
            var status = this.context.status;
            return status == _code.STATUS_DOLOGIN_USERGRID || status == _code.STATUS_DOLOGIN_IM;
        };

        /**
         * @private
         *
         */
        connection.prototype.isClosing = function () {
            return this.context.status == _code.STATUS_CLOSING;
        };

        /**
         * @private
         *
         */
        connection.prototype.isClosed = function () {
            return this.context.status == _code.STATUS_CLOSED;
        };

        /**
         * @private
         *
         */
        connection.prototype.clear = function () {
            var key = this.context.appKey;
            if (this.errorType != _code.WEBIM_CONNCTION_DISCONNECTED) {
                if (this.logOut) {
                    this.unSendMsgArr = [];
                    this.offLineSendConnecting = false;
                    this.context = {
                        status: _code.STATUS_INIT,
                        appKey: key
                    };
                }
            }
            if (this.intervalId) {
                clearInterval(this.intervalId);
            }
            this.restIndex = 0;
            this.xmppIndex = 0;

            if (this.errorType == _code.WEBIM_CONNCTION_CLIENT_LOGOUT || this.errorType == -1) {
                var message = {
                    data: {
                        data: "logout"
                    },
                    type: _code.WEBIM_CONNCTION_CLIENT_LOGOUT
                };
                this.onError(message);
            }
        };

        /**
         * 获取聊天室列表
         * @param {Object} options
         * @param {String} options.pagenum
         * @param {String} options.pagesize
         */
        connection.prototype.getChatRooms = function (options) {

            var conn = this,
                token = options.accessToken || this.context.accessToken;

            if (!_utils.isCanSetRequestHeader) {
                conn.onError({
                    type: _code.WEBIM_CONNCTION_NOT_SUPPORT_CHATROOM_ERROR
                });
                return;
            }

            if (token) {
                var apiUrl = this.apiUrl;
                var appName = this.context.appName;
                var orgName = this.context.orgName;

                if (!appName || !orgName) {
                    conn.onError({
                        type: _code.WEBIM_CONNCTION_AUTH_ERROR
                    });
                    return;
                }

                var suc = function suc(data, xhr) {
                    typeof options.success === 'function' && options.success(data);
                };

                var error = function error(res, xhr, msg) {
                    if (res.error && res.error_description) {
                        conn.onError({
                            type: _code.WEBIM_CONNCTION_LOAD_CHATROOM_ERROR,
                            msg: res.error_description,
                            data: res,
                            xhr: xhr
                        });
                    }
                };

                var pageInfo = {
                    pagenum: parseInt(options.pagenum) || 1,
                    pagesize: parseInt(options.pagesize) || 20
                };

                var opts = {
                    url: apiUrl + '/' + orgName + '/' + appName + '/chatrooms',
                    dataType: 'json',
                    type: 'GET',
                    headers: {'Authorization': 'Bearer ' + token},
                    data: pageInfo,
                    success: suc || _utils.emptyfn,
                    error: error || _utils.emptyfn
                };
                _utils.ajax(opts);
            } else {
                conn.onError({
                    type: _code.WEBIM_CONNCTION_TOKEN_NOT_ASSIGN_ERROR
                });
            }
        };

        /**
         * 加入聊天室
         * @param {Object} options
         * @param {String} options.roomId
         */
        connection.prototype.joinChatRoom = function (options) {
            var roomJid = this.context.appKey + '_' + options.roomId + '@conference.' + this.domain;
            var room_nick = roomJid + '/' + this.context.userId;
            var suc = options.success || _utils.emptyfn;
            var err = options.error || _utils.emptyfn;
            var errorFn = function errorFn(ele) {
                err({
                    type: _code.WEBIM_CONNCTION_JOINCHATROOM_ERROR,
                    data: ele
                });
            };

            var iq = $pres({
                from: this.context.jid,
                to: room_nick
            }).c('x', {xmlns: Strophe.NS.MUC + '#user'}).c('item', {
                affiliation: 'member',
                role: 'participant'
            }).up().up().c('roomtype', {xmlns: 'easemob:x:roomtype', type: 'chatroom'});

            this.context.stropheConn.sendIQ(iq.tree(), suc, errorFn);
        };

        /**
         * 退出聊天室
         * @param {Object} options
         * @param {String} options.roomId
         */
        connection.prototype.quitChatRoom = function (options) {
            var roomJid = this.context.appKey + '_' + options.roomId + '@conference.' + this.domain;
            var room_nick = roomJid + '/' + this.context.userId;
            var suc = options.success || _utils.emptyfn;
            var err = options.error || _utils.emptyfn;
            var errorFn = function errorFn(ele) {
                err({
                    type: _code.WEBIM_CONNCTION_QUITCHATROOM_ERROR,
                    data: ele
                });
            };
            var iq = $pres({
                from: this.context.jid,
                to: room_nick,
                type: 'unavailable'
            }).c('x', {xmlns: Strophe.NS.MUC + '#user'}).c('item', {
                affiliation: 'none',
                role: 'none'
            }).up().up().c('roomtype', {xmlns: 'easemob:x:roomtype', type: 'chatroom'});

            this.context.stropheConn.sendIQ(iq.tree(), suc, errorFn);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onReceiveInviteFromGroup = function (info) {
            info = eval('(' + info + ')');
            var self = this;
            var options = {
                title: "Group invitation",
                msg: info.user + " invites you to join into group:" + info.group_id,
                agree: function agree() {
                    WebIM.doQuery('{"type":"acceptInvitationFromGroup","id":"' + info.group_id + '","user":"' + info.user + '"}', function (response) {
                    }, function (code, msg) {
                        var message = {
                            data: {
                                data: "acceptInvitationFromGroup error:" + msg
                            },
                            type: _code.WEBIM_CONNECTION_ACCEPT_INVITATION_FROM_GROUP
                        };
                        self.onError(message);
                    });
                },
                reject: function reject() {
                    WebIM.doQuery('{"type":"declineInvitationFromGroup","id":"' + info.group_id + '","user":"' + info.user + '"}', function (response) {
                    }, function (code, msg) {
                        var message = {
                            data: {
                                data: "declineInvitationFromGroup error:" + msg
                            },
                            type: _code.WEBIM_CONNECTION_DECLINE_INVITATION_FROM_GROUP
                        };
                        self.onError(message);
                    });
                }
            };

            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onReceiveInviteAcceptionFromGroup = function (info) {
            info = eval('(' + info + ')');
            var options = {
                title: "Group invitation response",
                msg: info.user + " agreed to join into group:" + info.group_id,
                agree: function agree() {
                }
            };
            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onReceiveInviteDeclineFromGroup = function (info) {
            info = eval('(' + info + ')');
            var options = {
                title: "Group invitation response",
                msg: info.user + " rejected to join into group:" + info.group_id,
                agree: function agree() {
                }
            };
            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onAutoAcceptInvitationFromGroup = function (info) {
            info = eval('(' + info + ')');
            var options = {
                title: "Group invitation",
                msg: "You had joined into the group:" + info.group_name + " automatically.Inviter:" + info.user,
                agree: function agree() {
                }
            };
            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onLeaveGroup = function (info) {
            info = eval('(' + info + ')');
            var options = {
                title: "Group notification",
                msg: "You have been out of the group:" + info.group_id + ".Reason:" + info.msg,
                agree: function agree() {
                }
            };
            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onReceiveJoinGroupApplication = function (info) {
            info = eval('(' + info + ')');
            var self = this;
            var options = {
                title: "Group join application",
                msg: info.user + " applys to join into group:" + info.group_id,
                agree: function agree() {
                    WebIM.doQuery('{"type":"acceptJoinGroupApplication","id":"' + info.group_id + '","user":"' + info.user + '"}', function (response) {
                    }, function (code, msg) {
                        var message = {
                            data: {
                                data: "acceptJoinGroupApplication error:" + msg
                            },
                            type: _code.WEBIM_CONNECTION_ACCEPT_JOIN_GROUP
                        };
                        self.onError(message);
                    });
                },
                reject: function reject() {
                    WebIM.doQuery('{"type":"declineJoinGroupApplication","id":"' + info.group_id + '","user":"' + info.user + '"}', function (response) {
                    }, function (code, msg) {
                        var message = {
                            data: {
                                data: "declineJoinGroupApplication error:" + msg
                            },
                            type: _code.WEBIM_CONNECTION_DECLINE_JOIN_GROUP
                        };
                        self.onError(message);
                    });
                }
            };
            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onReceiveAcceptionFromGroup = function (info) {
            info = eval('(' + info + ')');
            var options = {
                title: "Group notification",
                msg: "You had joined into the group:" + info.group_name + ".",
                agree: function agree() {
                }
            };
            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onReceiveRejectionFromGroup = function () {
            info = eval('(' + info + ')');
            var options = {
                title: "Group notification",
                msg: "You have been rejected to join into the group:" + info.group_name + ".",
                agree: function agree() {
                }
            };
            this.onConfirmPop(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onUpdateMyGroupList = function (options) {
            this.onUpdateMyGroupList(options);
        };

        /**
         * for windowSDK
         * @private
         *
         */
        connection.prototype._onUpdateMyRoster = function (options) {
            this.onUpdateMyRoster(options);
        };

        /**
         * @private
         *
         */
        connection.prototype.reconnect = function () {
            var that = this;
            setTimeout(function () {
                _login(that.context.restTokenData, that);
            }, (this.autoReconnectNumTotal == 0 ? 0 : this.autoReconnectInterval) * 1000);
            this.autoReconnectNumTotal++;
        };

        /**
         *
         * @private
         * @deprecated
         */
        connection.prototype.closed = function () {
            var message = {
                data: {
                    data: "Closed error"
                },
                type: _code.WEBIM_CONNECTION_CLOSED
            };
            this.onError(message);
        };

        /**
         * 将消息序列化后存入localStorage
         * @param message {Object} 消息实体
         * @param type {String} 消息类型
         * @param status {String} 消息状态
         */
        connection.prototype.addToLocal = function (message, type, status) {
            if (!this.saveLocal) {
                return;
            }
            var sendByMe = typeof message.msg == 'string';
            if (!window.localStorage) return;
            try {
                var msg = _deepClone(message);
            } catch (e) {
                console.log(e.message);
            }
            msg.data = msg.sourceMsg;
            if (type == 'txt') {
                if (!message.data && !message.msg) {
                    return;
                }
                if (sendByMe) {
                    msg.data = this.enc(msg.msg);
                    delete msg.msg;
                } else {
                    msg.data = msg.sourceMsg;
                }
            }
            msg.msgType = type;
            msg.msgStatus = status;
            var oldRecord = window.localStorage.getItem(this.user);
            var serializedChatRecord = JSON.stringify(msg);
            if (oldRecord && (oldRecord.indexOf(message.id) >= 0 || oldRecord.indexOf(serializedChatRecord) >= 0)) {
                return;
            }
            var record = "";
            if (!oldRecord || oldRecord == "") {
                record = serializedChatRecord;
            } else {
                record = oldRecord + '\n' + serializedChatRecord;
            }
            window.localStorage.setItem(this.user, record);
        };

        /**
         * 将文本消息加密
         * @param messageSource {Object} 消息实体
         */

        connection.prototype.enc = function (messageSource) {
            var message = _.clone(messageSource);
            if (this.encrypt.type === 'base64') {
                message = btoa(messageSource);
            } else if (this.encrypt.type === 'aes') {
                var key = CryptoJS.enc.Utf8.parse(this.encrypt.key);
                var iv = CryptoJS.enc.Utf8.parse(this.encrypt.iv);
                var mode = this.encrypt.mode.toLowerCase();
                var option = {};
                if (mode === 'cbc') {
                    option = {
                        iv: iv,
                        mode: CryptoJS.mode.CBC,
                        padding: CryptoJS.pad.Pkcs7
                    };
                } else if (mode === 'ebc') {
                    option = {
                        mode: CryptoJS.mode.ECB,
                        padding: CryptoJS.pad.Pkcs7
                    };
                }
                var encryptedData = CryptoJS.AES.encrypt(message, key, option);

                message = encryptedData.toString();
            }
            return message;
        };

        /**
         * 将文本消息解密
         * @param source {Object} 消息实体
         * @returns {Object} 解密后的消息
         */
        connection.prototype.decrypt = function (source) {
            var receiveMsg = source,
                self = this;
            if (self.encrypt.type === 'base64') {
                receiveMsg = atob(receiveMsg);
            } else if (self.encrypt.type === 'aes') {
                var key = CryptoJS.enc.Utf8.parse(self.encrypt.key);
                var iv = CryptoJS.enc.Utf8.parse(self.encrypt.iv);
                var mode = self.encrypt.mode.toLowerCase();
                var option = {};
                if (mode === 'cbc') {
                    option = {
                        iv: iv,
                        mode: CryptoJS.mode.CBC,
                        padding: CryptoJS.pad.Pkcs7
                    };
                } else if (mode === 'ebc') {
                    option = {
                        mode: CryptoJS.mode.ECB,
                        padding: CryptoJS.pad.Pkcs7
                    };
                }
                var encryptedBase64Str = receiveMsg;
                var decryptedData = CryptoJS.AES.decrypt(encryptedBase64Str, key, option);
                var decryptedStr = decryptedData.toString(CryptoJS.enc.Utf8);
                receiveMsg = decryptedStr;
            }
            return receiveMsg;
        };

        /**
         * 从localStorage获取消息并反序列化
         * @returns {Array|*} 所有消息组成的数组
         */
        connection.prototype.getLocal = function () {
            if (!window.localStorage || !this.saveLocal) return;
            var user = this.user;
            var record = window.localStorage.getItem(user);

            if (!record || record == '') return;

            var recordArr = record.split('\n');
            for (var i in recordArr) {
                var recordItem = recordArr[i];
                recordItem = JSON.parse(recordItem);
                recordItem.data = this.decrypt(recordItem.data);
                recordArr[i] = recordItem;
            }
            return recordArr;
        };

        /**
         * used for blacklist
         * @private
         *
         */
        function _parsePrivacy(iq) {
            var list = [];
            var items = iq.getElementsByTagName('item');

            if (items) {
                for (var i = 0; i < items.length; i++) {
                    var item = items[i];
                    var jid = item.getAttribute('value');
                    var order = item.getAttribute('order');
                    var type = item.getAttribute('type');
                    if (!jid) {
                        continue;
                    }
                    var n = _parseNameFromJidFn(jid);
                    list[n] = {
                        type: type,
                        order: order,
                        jid: jid,
                        name: n
                    };
                }
            }
            return list;
        };

        /**
         * 获取好友黑名单
         *
         * @returns {Object} 好友列表
         */
        connection.prototype.getBlacklist = function (options) {
            options = options || {};
            var iq = $iq({type: 'get'});
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;
            var me = this;

            iq.c('query', {xmlns: 'jabber:iq:privacy'}).c('list', {name: 'special'});

            this.context.stropheConn.sendIQ(iq.tree(), function (iq) {
                me.onBlacklistUpdate(_parsePrivacy(iq));
                sucFn();
            }, function () {
                me.onBlacklistUpdate([]);
                errFn();
            });
        };

        /**
         * 将好友加入到黑名单
         * @param {Object} options
         * @param {Object[]} options.list - 调用这个函数后黑名单的所有名单列表，key值为好友的ID
         * @param {Object} options.list[].type=jid - 要加到黑名单的好友对象的type，默认是jid
         * @param {Number} options.list[].order - 要加到黑名单的好友对象的order，所有order不重复
         * @param {string} options.list[].jid - 要加到黑名单的好友的jid
         * @param {string} options.list[].name - 要加到黑名单的好友的ID
         */
        connection.prototype.addToBlackList = function (options) {
            var iq = $iq({type: 'set'});
            var blacklist = options.list || {};
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;
            var piece = iq.c('query', {xmlns: 'jabber:iq:privacy'}).c('list', {name: 'special'});

            var keys = Object.keys(blacklist);
            var len = keys.length;
            var order = 2;

            for (var i = 0; i < len; i++) {
                var item = blacklist[keys[i]];
                var type = item.type || 'jid';
                var jid = item.jid;

                piece = piece.c('item', {action: 'deny', order: order++, type: type, value: jid}).c('message');
                if (i !== len - 1) {
                    piece = piece.up().up();
                }
            }

            this.context.stropheConn.sendIQ(piece.tree(), sucFn, errFn);
        };

        /**
         * 将好友从黑名单移除
         * @param {Object} options
         * @param {Object[]} options.list - 调用这个函数后黑名单的所有名单列表，key值为好友的ID
         * @param {Object} options.list[].type=jid - 要加到黑名单的好友对象的type，默认是jid
         * @param {Number} options.list[].order - 要加到黑名单的好友对象的order，所有order不重复
         * @param {string} options.list[].jid - 要加到黑名单的好友的jid
         * @param {string} options.list[].name - 要加到黑名单的好友的ID
         */
        connection.prototype.removeFromBlackList = function (options) {
            console.log('removeFromBlackList: ', options);
            var iq = $iq({type: 'set'});
            var blacklist = options.list || {};
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;
            var piece = iq.c('query', {xmlns: 'jabber:iq:privacy'}).c('list', {name: 'special'});

            var keys = Object.keys(blacklist);
            var len = keys.length;

            for (var i = 0; i < len; i++) {
                var item = blacklist[keys[i]];
                var type = item.type || 'jid';
                var jid = item.jid;
                var order = item.order;

                piece = piece.c('item', {action: 'deny', order: order, type: type, value: jid}).c('message');
                if (i !== len - 1) {
                    piece = piece.up().up();
                }
            }

            this.context.stropheConn.sendIQ(piece.tree(), sucFn, errFn);
        };

        /**
         *
         * @private
         */
        connection.prototype._getGroupJid = function (to) {
            var appKey = this.context.appKey || '';
            return appKey + '_' + to + '@conference.' + this.domain;
        };

        /**
         * 加入群组黑名单
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.addToGroupBlackList = function (options) {
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;
            var jid = _getJid(options, this);
            var affiliation = 'admin'; //options.affiliation || 'admin';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'set', to: to});

            iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation}).c('item', {
                affiliation: 'outcast',
                jid: jid
            });

            this.context.stropheConn.sendIQ(iq.tree(), sucFn, errFn);
        };

        /**
         *
         * @private
         */
        function _parseGroupBlacklist(iq) {
            var list = {};
            var items = iq.getElementsByTagName('item');

            if (items) {
                for (var i = 0; i < items.length; i++) {
                    var item = items[i];
                    var jid = item.getAttribute('jid');
                    var affiliation = item.getAttribute('affiliation');
                    var nick = item.getAttribute('nick');
                    if (!jid) {
                        continue;
                    }
                    var n = _parseNameFromJidFn(jid);
                    list[n] = {
                        jid: jid,
                        affiliation: affiliation,
                        nick: nick,
                        name: n
                    };
                }
            }
            return list;
        }

        /**
         * 获取群组黑名单
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.getGroupBlacklist = function (options) {
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;

            // var jid = _getJid(options, this);
            var affiliation = 'admin'; //options.affiliation || 'admin';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'get', to: to});

            iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation}).c('item', {
                affiliation: 'outcast'
            });

            this.context.stropheConn.sendIQ(iq.tree(), function (msginfo) {
                sucFn(_parseGroupBlacklist(msginfo));
            }, function () {
                errFn();
            });
        };

        /**
         * 从群组黑名单删除
         * @param {Object} options
         * @deprecated
         */
        connection.prototype.removeGroupMemberFromBlacklist = function (options) {
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;

            var jid = _getJid(options, this);
            var affiliation = 'admin'; //options.affiliation || 'admin';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'set', to: to});

            iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation}).c('item', {
                affiliation: 'none',
                jid: jid
            });

            this.context.stropheConn.sendIQ(iq.tree(), function (msginfo) {
                sucFn();
            }, function () {
                errFn();
            });
        };

        /**
         * 修改群名称
         * @param {Object} options -
         * @deprecated
         */
        connection.prototype.changeGroupSubject = function (options) {
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;

            // must be `owner`
            var affiliation = 'owner';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'set', to: to});

            iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation}).c('x', {
                type: 'submit',
                xmlns: 'jabber:x:data'
            }).c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up().c('field', {'var': 'muc#roomconfig_roomname'}).c('value').t(options.subject).up().up().c('field', {'var': 'muc#roomconfig_roomdesc'}).c('value').t(options.description);

            this.context.stropheConn.sendIQ(iq.tree(), function (msginfo) {
                sucFn();
            }, function () {
                errFn();
            });
        };

        /**
         * 删除群组
         *
         * @param {Object} options -
         * @example
         <iq id="9BEF5D20-841A-4048-B33A-F3F871120E58" to="easemob-demo#chatdemoui_1477462231499@conference.easemob.com" type="set">
         <query xmlns="http://jabber.org/protocol/muc#owner">
         <destroy>
         <reason>xxx destory group yyy</reason>
         </destroy>
         </query>
         </iq>
         * @deprecated
         */
        connection.prototype.destroyGroup = function (options) {
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;

            // must be `owner`
            var affiliation = 'owner';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'set', to: to});

            iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation}).c('destroy').c('reason').t(options.reason || '');

            this.context.stropheConn.sendIQ(iq.tree(), function (msginfo) {
                sucFn();
            }, function () {
                errFn();
            });
        };

        /**
         * 主动离开群组
         *
         * @param {Object} options -
         * @example
         <iq id="5CD33172-7B62-41B7-98BC-CE6EF840C4F6_easemob_occupants_change_affiliation" to="easemob-demo#chatdemoui_1477481609392@conference.easemob.com" type="set">
         <query xmlns="http://jabber.org/protocol/muc#admin">
         <item affiliation="none" jid="easemob-demo#chatdemoui_lwz2@easemob.com"/>
         </query>
         </iq>
         <presence to="easemob-demo#chatdemoui_1479811172349@conference.easemob.com/mt002" type="unavailable"/>
         * @deprecated
         */
        connection.prototype.leaveGroupBySelf = function (options) {
            var self = this;
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;

            // must be `owner`
            var jid = _getJid(options, this);
            var affiliation = 'admin';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'set', to: to});

            iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation}).c('item', {
                affiliation: 'none',
                jid: jid
            });

            this.context.stropheConn.sendIQ(iq.tree(), function (msgInfo) {
                sucFn(msgInfo);
                var pres = $pres({type: 'unavailable', to: to + '/' + self.context.userId});
                self.sendCommand(pres.tree());
            }, function (errInfo) {
                errFn(errInfo);
            });
        };

        /**
         * 群主从群组中踢人，后续会改为调用RestFul API
         *
         * @param {Object} options -
         * @param {string[]} options.list - 需要从群组移除的好友ID组成的数组
         * @param {string} options.roomId - 群组ID
         * @deprecated
         * @example
         <iq id="9fb25cf4-1183-43c9-961e-9df70e300de4:sendIQ" to="easemob-demo#chatdemoui_1477481597120@conference.easemob.com" type="set" xmlns="jabber:client">
         <query xmlns="http://jabber.org/protocol/muc#admin">
         <item affiliation="none" jid="easemob-demo#chatdemoui_lwz4@easemob.com"/>
         <item jid="easemob-demo#chatdemoui_lwz4@easemob.com" role="none"/>
         <item affiliation="none" jid="easemob-demo#chatdemoui_lwz2@easemob.com"/>
         <item jid="easemob-demo#chatdemoui_lwz2@easemob.com" role="none"/>
         </query>
         </iq>
         */
        connection.prototype.leaveGroup = function (options) {
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;
            var list = options.list || [];
            var affiliation = 'admin';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'set', to: to});
            var piece = iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation});
            var keys = Object.keys(list);
            var len = keys.length;

            for (var i = 0; i < len; i++) {
                var name = list[keys[i]];
                var jid = _getJidByName(name, this);

                piece = piece.c('item', {
                    affiliation: 'none',
                    jid: jid
                }).up().c('item', {
                    role: 'none',
                    jid: jid
                }).up();
            }

            this.context.stropheConn.sendIQ(iq.tree(), function (msgInfo) {
                sucFn(msgInfo);
            }, function (errInfo) {
                errFn(errInfo);
            });
        };

        /**
         * 添加群组成员
         *
         * @param {Object} options -
         * @deprecated
         * @example
         Attention the sequence: message first (每个成员单独发一条message), iq second (多个成员可以合成一条iq发)
         <!-- 添加成员通知：send -->
         <message to='easemob-demo#chatdemoui_1477482739698@conference.easemob.com'>
         <x xmlns='http://jabber.org/protocol/muc#user'>
         <invite to='easemob-demo#chatdemoui_lwz2@easemob.com'>
         <reason>liuwz invite you to join group '谢谢'</reason>
         </invite>
         </x>
         </message>
         <!-- 添加成员：send -->
         <iq id='09DFB1E5-C939-4C43-B5A7-8000DA0E3B73_easemob_occupants_change_affiliation' to='easemob-demo#chatdemoui_1477482739698@conference.easemob.com' type='set'>
         <query xmlns='http://jabber.org/protocol/muc#admin'>
         <item affiliation='member' jid='easemob-demo#chatdemoui_lwz2@easemob.com'/>
         </query>
         </iq>
         */
        connection.prototype.addGroupMembers = function (options) {
            var sucFn = options.success || _utils.emptyfn;
            var errFn = options.error || _utils.emptyfn;
            var list = options.list || [];
            var affiliation = 'admin';
            var to = this._getGroupJid(options.roomId);
            var iq = $iq({type: 'set', to: to});
            var piece = iq.c('query', {xmlns: 'http://jabber.org/protocol/muc#' + affiliation});
            var len = list.length;

            for (var i = 0; i < len; i++) {

                var name = list[i];
                var jid = _getJidByName(name, this);

                piece = piece.c('item', {
                    affiliation: 'member',
                    jid: jid
                }).up();

                var dom = $msg({
                    to: to
                }).c('x', {
                    xmlns: 'http://jabber.org/protocol/muc#user'
                }).c('invite', {
                    to: jid
                }).c('reason').t(options.reason || '');

                this.sendCommand(dom.tree());
            }

            this.context.stropheConn.sendIQ(iq.tree(), function (msgInfo) {
                sucFn(msgInfo);
            }, function (errInfo) {
                errFn(errInfo);
            });
        };

        /**
         * 接受加入申请
         *
         * @param {Object} options -
         * @deprecated
         */
        connection.prototype.acceptInviteFromGroup = function (options) {
            options.success = function () {
                // then send sendAcceptInviteMessage
                // connection.prototype.sendAcceptInviteMessage(optoins);
            };
            this.addGroupMembers(options);
        };

        /**
         * 拒绝入群申请
         * @param {Object} options -
         * @example
         throw request for now 暂时不处理，直接丢弃

         <message to='easemob-demo#chatdemoui_mt002@easemob.com' from='easmeob-demo#chatdemoui_mt001@easemob.com' id='B83B7210-BCFF-4DEE-AB28-B9FE5579C1E2'>
         <x xmlns='http://jabber.org/protocol/muc#user'>
         <apply to='easemob-demo#chatdemoui_groupid1@conference.easemob.com' from='easmeob-demo#chatdemoui_mt001@easemob.com' toNick='llllll'>
         <reason>reject</reason>
         </apply>
         </x>
         </message>
         * @deprecated
         */
        connection.prototype.rejectInviteFromGroup = function (options) {
            // var from = _getJidByName(options.from, this);
            // var dom = $msg({
            //     from: from,
            //     to: _getJidByName(options.to, this)
            // }).c('x', {
            //     xmlns: 'http://jabber.org/protocol/muc#user'
            // }).c('apply', {
            //     from: from,
            //     to: this._getGroupJid(options.groupId),
            //     toNick: options.groupName
            // }).c('reason').t(options.reason || '');
            //
            // this.sendCommand(dom.tree());
        };

        /**
         * 创建群组-异步
         * @param {Object} p -
         * @deprecated
         */
        connection.prototype.createGroupAsync = function (p) {
            var roomId = p.from;
            var me = this;
            var toRoom = this._getGroupJid(roomId);
            var to = toRoom + '/' + this.context.userId;
            var options = this.groupOption;
            var suc = p.success || _utils.emptyfn;

            // Creating a Reserved Room
            var iq = $iq({type: 'get', to: toRoom}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});

            // Strophe.info('step 1 ----------');
            // Strophe.info(options);
            me.context.stropheConn.sendIQ(iq.tree(), function (msgInfo) {
                // console.log(msgInfo);

                // for ie hack
                if ('setAttribute' in msgInfo) {
                    // Strophe.info('step 3 ----------');
                    var x = msgInfo.getElementsByTagName('x')[0];
                    x.setAttribute('type', 'submit');
                } else {
                    // Strophe.info('step 4 ----------');
                    Strophe.forEachChild(msgInfo, 'x', function (field) {
                        field.setAttribute('type', 'submit');
                    });
                }

                Strophe.info('step 5 ----------');
                Strophe.forEachChild(x, 'field', function (field) {
                    var fieldVar = field.getAttribute('var');
                    var valueDom = field.getElementsByTagName('value')[0];
                    Strophe.info(fieldVar);
                    switch (fieldVar) {
                        case 'muc#roomconfig_maxusers':
                            _setText(valueDom, options.optionsMaxUsers || 200);
                            break;
                        case 'muc#roomconfig_roomname':
                            _setText(valueDom, options.subject || '');
                            break;
                        case 'muc#roomconfig_roomdesc':
                            _setText(valueDom, options.description || '');
                            break;
                        case 'muc#roomconfig_publicroom':
                            // public 1
                            _setText(valueDom, +options.optionsPublic);
                            break;
                        case 'muc#roomconfig_membersonly':
                            _setText(valueDom, +options.optionsMembersOnly);
                            break;
                        case 'muc#roomconfig_moderatedroom':
                            _setText(valueDom, +options.optionsModerate);
                            break;
                        case 'muc#roomconfig_persistentroom':
                            _setText(valueDom, 1);
                            break;
                        case 'muc#roomconfig_allowinvites':
                            _setText(valueDom, +options.optionsAllowInvites);
                            break;
                        case 'muc#roomconfig_allowvisitornickchange':
                            _setText(valueDom, 0);
                            break;
                        case 'muc#roomconfig_allowvisitorstatus':
                            _setText(valueDom, 0);
                            break;
                        case 'allow_private_messages':
                            _setText(valueDom, 0);
                            break;
                        case 'allow_private_messages_from_visitors':
                            _setText(valueDom, 'nobody');
                            break;
                        default:
                            break;
                    }
                });

                var iq = $iq({
                    to: toRoom,
                    type: 'set'
                }).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}).cnode(x);

                me.context.stropheConn.sendIQ(iq.tree(), function (msgInfo) {
                    me.addGroupMembers({
                        list: options.members,
                        roomId: roomId
                    });

                    suc(options);
                }, function (errInfo) {
                    // errFn(errInfo);
                });
                // sucFn(msgInfo);
            }, function (errInfo) {
                // errFn(errInfo);
            });
        };

        /**
         * 创建群组
         * @param {Object} options -
         * @deprecated
         * @example
         * 1. 创建申请 -> 得到房主身份
         * 2. 获取房主信息 -> 得到房间form
         * 3. 完善房间form -> 创建成功
         * 4. 添加房间成员
         * 5. 消息通知成员
         */
        connection.prototype.createGroup = function (options) {
            this.groupOption = options;
            var roomId = +new Date();
            var toRoom = this._getGroupJid(roomId);
            var to = toRoom + '/' + this.context.userId;

            var pres = $pres({to: to}).c('x', {xmlns: 'http://jabber.org/protocol/muc'}).up().c('create', {xmlns: 'http://jabber.org/protocol/muc'}).up();

            // createGroupACK
            this.sendCommand(pres.tree());
        };

        /**
         * 通过RestFul API接口创建群组
         * @param opt {Object} - 群组信息
         * @param opt.data.groupname {string} - 群组名
         * @param opt.data.desc {string} - 群组描述
         * @param opt.data.members {string[]} - 群好友列表
         * @param opt.data.public {Boolean} - true: 公开群，false: 私有群
         * @param opt.data.approval {Boolean} - 前提：opt.data.public=true, true: 加群需要审批，false: 加群无需审批
         * @param opt.data.allowinvites {Boolean} - 前提：opt.data.public=false, true: 允许成员邀请入群，false: 不允许成员邀请入群
         * @since 1.4.11
         */
        connection.prototype.createGroupNew = function (opt) {
            opt.data.owner = "" + this.user + "";
            opt.data.maxusers = parseInt(opt.data.maxusers);
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/chatgroups',
                dataType: 'json',
                type: 'POST',
                data: JSON.stringify(opt.data),
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = function (respData) {
                opt.success(respData);
                this.onCreateGroup(respData);
            }.bind(this);
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API屏蔽群组，只对移动端有效
         * @param {Object} options -
         * @param {string} options.groupId - 需要屏蔽的群组ID
         * @since 1.4.11
         */
        connection.prototype.blockGroup = function (opt) {
            var groupId = opt.groupId;
            groupId = 'notification_ignore_' + groupId;
            var data = {
                entities: []
            };
            data.entities[0] = {};
            data.entities[0][groupId] = true;
            var options = {
                type: 'PUT',
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'users' + '/' + this.user,
                data: JSON.stringify(data),
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API发出入群申请
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @since 1.4.11
         */
        connection.prototype.joinGroup = function (opt) {
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + opt.groupId + '/' + 'apply',
                type: 'POST',
                dataType: 'json',
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API分页获取群组列表
         * @param {Object} opt -
         * @param {Number} opt.limit - 每一页群组的最大数目
         * @param {string} opt.cursor=null - 游标，如果数据还有下一页，API 返回值会包含此字段，传递此字段可获取下一页的数据，为null时获取第一页数据
         * @since 1.4.11
         */
        connection.prototype.listGroups = function (opt) {
            var requestData = [];
            requestData['limit'] = opt.limit;
            requestData['cursor'] = opt.cursor;
            if (!requestData['cursor']) delete requestData['cursor'];
            if (isNaN(opt.limit)) {
                throw 'The parameter \"limit\" should be a number';
                return;
            }
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/publicchatgroups',
                type: 'GET',
                dataType: 'json',
                data: requestData,
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API根据groupId获取群组详情
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @since 1.4.11
         */
        connection.prototype.getGroupInfo = function (opt) {
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/chatgroups/' + opt.groupId,
                type: 'GET',
                dataType: 'json',
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API列出某用户所加入的所有群组
         * @param {Object} opt - 加入两个回调函数即可，success, error
         * @since 1.4.11
         */
        connection.prototype.getGroup = function (opt) {
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'users' + '/' + this.user + '/' + 'joined_chatgroups',
                dataType: 'json',
                type: 'GET',
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API分页列出群组的所有成员
         * @param {Object} opt -
         * @param {Number} opt.pageNum - 页码
         * @param {Number} opt.pageSize - 每一页的最大群成员数目
         * @param {string} opt.groupId - 群ID
         */
        connection.prototype.listGroupMember = function (opt) {
            if (isNaN(opt.pageNum) || opt.pageNum <= 0) {
                throw 'The parameter \"pageNum\" should be a positive number';
                return;
            } else if (isNaN(opt.pageSize) || opt.pageSize <= 0) {
                throw 'The parameter \"pageSize\" should be a positive number';
                return;
            } else if (opt.groupId === null && typeof opt.groupId === 'undefined') {
                throw 'The parameter \"groupId\" should be added';
                return;
            }
            var requestData = [],
                groupId = opt.groupId;
            requestData['pagenum'] = opt.pageNum;
            requestData['pagesize'] = opt.pageSize;
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/chatgroups' + '/' + groupId + '/users',
                dataType: 'json',
                type: 'GET',
                data: requestData,
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API禁止群用户发言
         * @param {Object} opt -
         * @param {string} opt.username - 被禁言的群成员的ID
         * @param {Number} opt.muteDuration - 被禁言的时长
         * @param {string} opt.groupId - 群ID
         * @since 1.4.11
         */
        connection.prototype.mute = function (opt) {
            var groupId = opt.groupId,
                requestData = {
                    "usernames": [opt.username],
                    "mute_duration": opt.muteDuration
                },
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'mute',
                    dataType: 'json',
                    type: 'POST',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    },
                    data: JSON.stringify(requestData)
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API取消对用户禁言
         * @param {Object} opt -
         * @param {string} opt.groupId - 群ID
         * @param {string} opt.username - 被取消禁言的群用户ID
         * @since 1.4.11
         */
        connection.prototype.removeMute = function (opt) {
            var groupId = opt.groupId,
                username = opt.username;
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'mute' + '/' + username,
                dataType: 'json',
                type: 'DELETE',
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API获取群组下所有管理员
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @since 1.4.11
         */
        connection.prototype.getGroupAdmin = function (opt) {
            var groupId = opt.groupId;
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/chatgroups' + '/' + groupId + '/admin',
                dataType: 'json',
                type: 'GET',
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API获取群组下所有被禁言成员
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         */
        connection.prototype.getMuted = function (opt) {
            var groupId = opt.groupId;
            var options = {
                url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/chatgroups' + '/' + groupId + '/mute',
                dataType: 'json',
                type: 'GET',
                headers: {
                    'Authorization': 'Bearer ' + this.token,
                    'Content-Type': 'application/json'
                }
            };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API设置群管理员
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @param {string} opt.username - 用户ID
         */
        connection.prototype.setAdmin = function (opt) {
            var groupId = opt.groupId,
                requestData = {
                    newadmin: opt.username
                },
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'admin',
                    type: "POST",
                    dataType: 'json',
                    data: JSON.stringify(requestData),
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API取消群管理员
         * @param {Object} opt -
         * @param {string} opt.gorupId - 群组ID
         * @param {string} opt.username - 用户ID
         */
        connection.prototype.removeAdmin = function (opt) {
            var groupId = opt.groupId,
                username = opt.username,
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'admin' + '/' + username,
                    type: 'DELETE',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API同意用户加入群
         * @param {Object} opt -
         * @param {string} opt.applicant - 申请加群的用户名
         * @param {Object} opt.groupId - 群组ID
         */
        connection.prototype.agreeJoinGroup = function (opt) {
            var groupId = opt.groupId,
                requestData = {
                    "applicant": opt.applicant,
                    "verifyResult": true,
                    "reason": "no clue"
                },
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'apply_verify',
                    type: 'POST',
                    dataType: "json",
                    data: JSON.stringify(requestData),
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API拒绝用户加入群
         * @param {Object} opt -
         * @param {string} opt.applicant - 申请加群的用户名
         * @param {Object} opt.groupId - 群组ID
         */
        connection.prototype.rejectJoinGroup = function (opt) {
            var groupId = opt.groupId,
                requestData = {
                    "applicant": opt.applicant,
                    "verifyResult": false,
                    "reason": "no clue"
                },
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'apply_verify',
                    type: 'POST',
                    dataType: "json",
                    data: JSON.stringify(requestData),
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API添加用户至群组黑名单(单个)
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @param {stirng} opt.username - 用户ID
         */
        connection.prototype.groupBlockSingle = function (opt) {
            var groupId = opt.groupId,
                username = opt.username,
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'blocks' + '/' + 'users' + '/' + username,
                    type: 'POST',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API添加用户至群组黑名单(批量)
         * @param {Object} opt -
         * @param {string[]} opt.username - 用户ID数组
         * @param {string} opt.groupId - 群组ID
         */
        connection.prototype.groupBlockMulti = function (opt) {
            var groupId = opt.groupId,
                usernames = opt.usernames,
                requestData = {
                    usernames: usernames
                },
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'blocks' + '/' + 'users',
                    data: JSON.stringify(requestData),
                    type: 'POST',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API将用户从群黑名单移除（单个）
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @param {string} opt.username - 用户名
         */
        connection.prototype.removeGroupBlockSingle = function (opt) {
            var groupId = opt.groupId,
                username = opt.username,
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'blocks' + '/' + 'users' + '/' + username,
                    type: 'DELETE',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API将用户从群黑名单移除（批量）
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组名
         * @param {string[]} opt.username - 用户ID数组
         */
        connection.prototype.removeGroupBlockMulti = function (opt) {
            var groupId = opt.groupId,
                username = opt.username.join(','),
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'blocks' + '/' + 'users' + '/' + username,
                    type: 'DELETE',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API解散群组
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         */
        connection.prototype.dissolveGroup = function (opt) {
            var groupId = opt.groupId,
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '?version=v3',
                    type: 'DELETE',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API获取群组黑名单
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         */
        connection.prototype.getGroupBlacklistNew = function (opt) {
            var groupId = opt.groupId,
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'blocks' + '/' + 'users',
                    type: 'GET',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API离开群组
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         */
        connection.prototype.quitGroup = function (opt) {
            var groupId = opt.groupId,
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'quit',
                    type: 'DELETE',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API修改群信息
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @param {string} opt.groupName - 群组名
         * @param {string} opt.description - 群组简介
         */
        connection.prototype.modifyGroup = function (opt) {
            var groupId = opt.groupId,
                requestData = {
                    groupname: opt.groupName,
                    description: opt.description
                },
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId,
                    type: 'PUT',
                    data: JSON.stringify(requestData),
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API删除单个群成员
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @param {string} opt.username - 用户名
         */
        connection.prototype.removeSingleGroupMember = function (opt) {
            var groupId = opt.groupId,
                username = opt.username,
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'users' + '/' + username,
                    type: 'DELETE',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API删除多个群成员
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组ID
         * @param {string[]} opt.users - 用户ID数组
         */
        connection.prototype.removeMultiGroupMember = function (opt) {
            var groupId = opt.groupId,
                users = opt.users.join(','),
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'users' + '/' + users,
                    type: 'DELETE',
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        /**
         * 通过RestFul API邀请群成员
         * @param {Object} opt -
         * @param {string} opt.groupId - 群组名
         * @param {string[]} opt.users - 用户名ID数组
         */
        connection.prototype.inviteToGroup = function (opt) {
            var groupId = opt.groupId,
                users = opt.users,
                requestData = {
                    usernames: users
                },
                options = {
                    url: this.apiUrl + '/' + this.orgName + '/' + this.appName + '/' + 'chatgroups' + '/' + groupId + '/' + 'invite',
                    type: 'POST',
                    data: JSON.stringify(requestData),
                    dataType: 'json',
                    headers: {
                        'Authorization': 'Bearer ' + this.token,
                        'Content-Type': 'application/json'
                    }
                };
            options.success = opt.success || _utils.emptyfn;
            options.error = opt.error || _utils.emptyfn;
            WebIM.utils.ajax(options);
        };

        function _setText(valueDom, v) {
            if ('textContent' in valueDom) {
                valueDom.textContent = v;
            } else if ('text' in valueDom) {
                valueDom.text = v;
            } else {
                // Strophe.info('_setText 4 ----------');
                // valueDom.innerHTML = v;
            }
        }

        var WebIM = window.WebIM || {};
        WebIM.connection = connection;
        WebIM.utils = _utils;
        WebIM.statusCode = _code;
        WebIM.message = _msg.message;
        WebIM.doQuery = function (str, suc, fail) {
            if (typeof window.cefQuery === 'undefined') {
                return;
            }
            window.cefQuery({
                request: str,
                persistent: false,
                onSuccess: suc,
                onFailure: fail
            });
        };

        /**************************** debug ****************************/
        function logMessage(message) {
            WebIM && WebIM.config.isDebug && console.log(WebIM.utils.ts() + '[recv] ', message.data);
        }

        if (WebIM && WebIM.config.isDebug) {
            Strophe.Connection.prototype.rawOutput = function (data) {
                console.log('%c ' + WebIM.utils.ts() + '[send] ' + data, "background-color: #e2f7da");
            };
        }

        if (WebIM && WebIM.config && WebIM.config.isSandBox) {
            WebIM.config.xmppURL = WebIM.config.xmppURL.replace('.easemob.', '-sandbox.easemob.');
            WebIM.config.apiURL = WebIM.config.apiURL.replace('.easemob.', '-sdb.easemob.');
        }

        module.exports = WebIM;

        if (false) {
            module.hot.accept();
        }

        /***/
    },

    /***/ 249:
    /***/ function (module, exports, __webpack_require__) {

        'use strict';

        var CryptoJS = __webpack_require__(211);
        ;(function () {
            'use strict';

            var _utils = __webpack_require__(206).utils;
            var Message = function Message(type, id) {
                if (!this instanceof Message) {
                    return new Message(type);
                }

                this._msg = {};

                if (typeof Message[type] === 'function') {
                    Message[type].prototype.setGroup = this.setGroup;
                    this._msg = new Message[type](id);
                }
                return this._msg;
            };
            Message.prototype.setGroup = function (group) {
                this.body.group = group;
            };

            /*
	     * Read Message
	     */
            Message.read = function (id) {
                this.id = id;
                this.type = 'read';
            };

            Message.read.prototype.set = function (opt) {
                this.body = {
                    ackId: opt.id,
                    to: opt.to
                };
            };

            /*
	     * deliver message
	     */
            Message.delivery = function (id) {
                this.id = id;
                this.type = 'delivery';
            };

            Message.delivery.prototype.set = function (opt) {
                this.body = {
                    bodyId: opt.id,
                    to: opt.to
                };
            };

            /*
	     * text message
	     */
            Message.txt = function (id) {
                this.id = id;
                this.type = 'txt';
                this.body = {};
            };
            Message.txt.prototype.set = function (opt) {
                this.value = opt.msg;
                this.body = {
                    id: this.id,
                    to: opt.to,
                    msg: this.value,
                    type: this.type,
                    roomType: opt.roomType,
                    ext: opt.ext || {},
                    success: opt.success,
                    fail: opt.fail
                };

                !opt.roomType && delete this.body.roomType;
            };

            /*
	     * cmd message
	     */
            Message.cmd = function (id) {
                this.id = id;
                this.type = 'cmd';
                this.body = {};
            };
            Message.cmd.prototype.set = function (opt) {
                this.value = '';

                this.body = {
                    to: opt.to,
                    action: opt.action,
                    msg: this.value,
                    type: this.type,
                    roomType: opt.roomType,
                    ext: opt.ext || {},
                    success: opt.success
                };
                !opt.roomType && delete this.body.roomType;
            };

            /*
	     * loc message
	     */
            Message.location = function (id) {
                this.id = id;
                this.type = 'loc';
                this.body = {};
            };
            Message.location.prototype.set = function (opt) {
                this.body = {
                    to: opt.to,
                    type: this.type,
                    roomType: opt.roomType,
                    addr: opt.addr,
                    lat: opt.lat,
                    lng: opt.lng,
                    ext: opt.ext || {}
                };
            };

            /*
	     * img message
	     */
            Message.img = function (id) {
                this.id = id;
                this.type = 'img';
                this.body = {};
            };
            Message.img.prototype.set = function (opt) {
                opt.file = opt.file || _utils.getFileUrl(opt.fileInputId);

                this.value = opt.file;

                this.body = {
                    id: this.id,
                    file: this.value,
                    apiUrl: opt.apiUrl,
                    to: opt.to,
                    type: this.type,
                    ext: opt.ext || {},
                    roomType: opt.roomType,
                    onFileUploadError: opt.onFileUploadError,
                    onFileUploadComplete: opt.onFileUploadComplete,
                    success: opt.success,
                    fail: opt.fail,
                    flashUpload: opt.flashUpload,
                    width: opt.width,
                    height: opt.height,
                    body: opt.body,
                    uploadError: opt.uploadError,
                    uploadComplete: opt.uploadComplete
                };

                !opt.roomType && delete this.body.roomType;
            };

            /*
	     * audio message
	     */
            Message.audio = function (id) {
                this.id = id;
                this.type = 'audio';
                this.body = {};
            };
            Message.audio.prototype.set = function (opt) {
                opt.file = opt.file || _utils.getFileUrl(opt.fileInputId);

                this.value = opt.file;
                this.filename = opt.filename || this.value.filename;

                this.body = {
                    id: this.id,
                    file: this.value,
                    filename: this.filename,
                    apiUrl: opt.apiUrl,
                    to: opt.to,
                    type: this.type,
                    ext: opt.ext || {},
                    length: opt.length || 0,
                    roomType: opt.roomType,
                    file_length: opt.file_length,
                    onFileUploadError: opt.onFileUploadError,
                    onFileUploadComplete: opt.onFileUploadComplete,
                    success: opt.success,
                    fail: opt.fail,
                    flashUpload: opt.flashUpload,
                    body: opt.body
                };
                !opt.roomType && delete this.body.roomType;
            };

            /*
	     * file message
	     */
            Message.file = function (id) {
                this.id = id;
                this.type = 'file';
                this.body = {};
            };
            Message.file.prototype.set = function (opt) {
                opt.file = opt.file || _utils.getFileUrl(opt.fileInputId);

                this.value = opt.file;
                this.filename = opt.filename || this.value.filename;

                this.body = {
                    id: this.id,
                    file: this.value,
                    filename: this.filename,
                    apiUrl: opt.apiUrl,
                    to: opt.to,
                    type: this.type,
                    ext: opt.ext || {},
                    roomType: opt.roomType,
                    onFileUploadError: opt.onFileUploadError,
                    onFileUploadComplete: opt.onFileUploadComplete,
                    success: opt.success,
                    fail: opt.fail,
                    flashUpload: opt.flashUpload,
                    body: opt.body
                };
                !opt.roomType && delete this.body.roomType;
            };

            /*
	     * video message
	     */
            Message.video = function (id) {
            };
            Message.video.prototype.set = function (opt) {
            };

            var _Message = function _Message(message) {

                if (!this instanceof _Message) {
                    return new _Message(message, conn);
                }

                this.msg = message;
            };

            _Message.prototype.send = function (conn) {
                var me = this;

                var _send = function _send(message) {

                    message.ext = message.ext || {};
                    message.ext.weichat = message.ext.weichat || {};
                    message.ext.weichat.originType = message.ext.weichat.originType || 'webim';

                    var dom;
                    var json = {
                        from: conn.context.userId || '',
                        to: message.to,
                        bodies: [message.body],
                        ext: message.ext || {}
                    };
                    var jsonstr = _utils.stringify(json);
                    dom = $msg({
                        type: message.group || 'chat',
                        to: message.toJid,
                        id: message.id,
                        xmlns: 'jabber:client'
                    }).c('body').t(jsonstr);

                    if (message.roomType) {
                        dom.up().c('roomtype', {xmlns: 'easemob:x:roomtype', type: 'chatroom'});
                    }
                    if (message.bodyId) {
                        dom = $msg({
                            from: conn.context.jid || '',
                            to: message.toJid,
                            id: message.id,
                            xmlns: 'jabber:client'
                        }).c('body').t(message.bodyId);
                        var delivery = {
                            xmlns: 'urn:xmpp:receipts',
                            id: message.bodyId
                        };
                        dom.up().c('delivery', delivery);
                    }
                    if (message.ackId) {

                        if (conn.context.jid.indexOf(message.toJid) >= 0) {
                            return;
                        }
                        dom = $msg({
                            from: conn.context.jid || '',
                            to: message.toJid,
                            id: message.id,
                            xmlns: 'jabber:client'
                        }).c('body').t(message.ackId);
                        var read = {
                            xmlns: 'urn:xmpp:receipts',
                            id: message.ackId
                        };
                        dom.up().c('acked', read);
                    }

                    setTimeout(function () {
                        if (typeof _msgHash !== 'undefined' && _msgHash[message.id]) {
                            _msgHash[message.id].msg.fail instanceof Function && _msgHash[message.id].msg.fail(message.id);
                        }
                    }, 60000);
                    conn.sendCommand(dom.tree(), message.id);
                };

                if (me.msg.file) {
                    if (me.msg.body && me.msg.body.url) {
                        // Only send msg
                        _send(me.msg);
                        return;
                    }
                    var _tmpComplete = me.msg.onFileUploadComplete;
                    var _complete = function _complete(data) {
                        if (data.entities[0]['file-metadata']) {
                            var file_len = data.entities[0]['file-metadata']['content-length'];
                            // me.msg.file_length = file_len;
                            me.msg.filetype = data.entities[0]['file-metadata']['content-type'];
                            if (file_len > 204800) {
                                me.msg.thumbnail = true;
                            }
                        }

                        me.msg.body = {
                            type: me.msg.type || 'file',

                            url: (location.protocol != 'https:' && conn.isHttpDNS ? conn.apiUrl + data.uri.substr(data.uri.indexOf("/", 9)) : data.uri) + '/' + data.entities[0]['uuid'],
                            secret: data.entities[0]['share-secret'],
                            filename: me.msg.file.filename || me.msg.filename,
                            size: {
                                width: me.msg.width || 0,
                                height: me.msg.height || 0
                            },
                            length: me.msg.length || 0,
                            file_length: me.msg.ext.file_length || 0,
                            filetype: me.msg.filetype
                        };
                        _send(me.msg);
                        _tmpComplete instanceof Function && _tmpComplete(data, me.msg.id);
                    };

                    me.msg.onFileUploadComplete = _complete;
                    _utils.uploadFile.call(conn, me.msg);
                } else {
                    me.msg.body = {
                        type: me.msg.type === 'chat' ? 'txt' : me.msg.type,
                        msg: me.msg.msg
                    };
                    if (me.msg.type === 'cmd') {
                        me.msg.body.action = me.msg.action;
                    } else if (me.msg.type === 'loc') {
                        me.msg.body.addr = me.msg.addr;
                        me.msg.body.lat = me.msg.lat;
                        me.msg.body.lng = me.msg.lng;
                    }

                    _send(me.msg);
                }
            };

            exports._msg = _Message;
            exports.message = Message;
        })();

        /***/
    },

    /***/ 250:
    /***/ function (module, exports) {

        "use strict";

        ;(function () {
            function Array_h(length) {
                this.array = length === undefined ? [] : new Array(length);
            }

            Array_h.prototype = {
                /**
                 * 返回数组长度
                 *
                 * @return {Number} length [数组长度]
                 */
                length: function length() {
                    return this.array.length;
                },

                at: function at(index) {
                    return this.array[index];
                },

                set: function set(index, obj) {
                    this.array[index] = obj;
                },

                /**
                 * 向数组的末尾添加一个或多个元素，并返回新的长度。
                 *
                 * @param  {*} obj [description]
                 * @return {Number} length [新数组的长度]
                 */
                push: function push(obj) {
                    return this.array.push(obj);
                },

                /**
                 * 返回数组中选定的元素
                 *
                 * @param  {Number} start [开始索引值]
                 * @param  {Number} end [结束索引值]
                 * @return {Array} newArray  [新的数组]
                 */
                slice: function slice(start, end) {
                    return this.array = this.array.slice(start, end);
                },

                concat: function concat(array) {
                    this.array = this.array.concat(array);
                },

                remove: function remove(index, count) {
                    count = count === undefined ? 1 : count;
                    this.array.splice(index, count);
                },

                join: function join(separator) {
                    return this.array.join(separator);
                },

                clear: function clear() {
                    this.array.length = 0;
                }
            };

            /**
             * 先进先出队列 (First Input First Output)
             *
             * 一种先进先出的数据缓存器
             */
            var Queue = function Queue() {
                this._array_h = new Array_h();
            };

            Queue.prototype = {
                _index: 0,

                /**
                 * 排队
                 *
                 * @param  {Object} obj [description]
                 * @return {[type]}     [description]
                 */
                push: function push(obj) {
                    this._array_h.push(obj);
                },

                /**
                 * 出队
                 *
                 * @return {Object} [description]
                 */
                pop: function pop() {
                    var ret = null;
                    if (this._array_h.length()) {
                        ret = this._array_h.at(this._index);
                        if (++this._index * 2 >= this._array_h.length()) {
                            this._array_h.slice(this._index);
                            this._index = 0;
                        }
                    }
                    return ret;
                },

                /**
                 * 返回队列中头部(即最新添加的)的动态对象
                 *
                 * @return {Object} [description]
                 */
                head: function head() {
                    var ret = null,
                        len = this._array_h.length();
                    if (len) {
                        ret = this._array_h.at(len - 1);
                    }
                    return ret;
                },

                /**
                 * 返回队列中尾部(即最早添加的)的动态对象
                 *
                 * @return {Object} [description]
                 */
                tail: function tail() {
                    var ret = null,
                        len = this._array_h.length();
                    if (len) {
                        ret = this._array_h.at(this._index);
                    }
                    return ret;
                },

                /**
                 * 返回数据队列长度
                 *
                 * @return {Number} [description]
                 */
                length: function length() {
                    return this._array_h.length() - this._index;
                },

                /**
                 * 队列是否为空
                 *
                 * @return {Boolean} [description]
                 */
                empty: function empty() {
                    return this._array_h.length() === 0;
                },

                clear: function clear() {
                    this._array_h.clear();
                }
            };
            exports.Queue = Queue;
        })();

        /***/
    }

    /******/
});